Temperature Variation
Water temperatures in Lake Champlain reached a minimum in February.
Sampling for this project began during the Spring warming period.
Temperature variability (both ranges and variance) increase with
temperature, but are strongly affected by the length of time period
examined.
# Lake Champlain near Burlington, VT
siteNumber = "04294500"
ChamplainInfo = readNWISsite(siteNumber)
parameterCd = "00010"
startDate = "2023-01-01"
endDate = ""
#statCd = c("00001", "00002","00003", "00011") # 1 - max, 2 - min, 3 = mean
# Constructs the URL for the data wanted then downloads the data
url = constructNWISURL(siteNumbers = siteNumber, parameterCd = parameterCd,
startDate = startDate, endDate = endDate, service = "uv")
temp_data = importWaterML1(url, asDateTime = T) %>%
mutate("date" = as.Date(dateTime)) %>%
select(date, "temp" = X_00010_00000)
## Daily values
daily_temp_data = temp_data %>%
ungroup() %>%
group_by(date) %>%
summarise(mean_temp = mean(temp),
med_temp = median(temp),
var_temp = var(temp),
min_temp = min(temp),
max_temp = max(temp)) %>%
mutate("range_temp" = max_temp - min_temp)
day_prior_temp_data = temp_data %>%
ungroup() %>%
group_by(date) %>%
summarise(mean_temp = mean(temp),
med_temp = median(temp),
var_temp = var(temp),
min_temp = min(temp),
max_temp = max(temp)) %>%
mutate(date = date + 1) %>%
rename_with(.fn = ~ paste0("prior_day_", .x), .cols = c(-date))
daily_plot = daily_temp_data %>%
pivot_longer(cols = c(-date),
names_to = "parameter",
values_to = "temp") %>%
ggplot(aes(x = date, y = temp, colour = parameter)) +
geom_line(linewidth = 1) +
scale_colour_manual(values = c(
"mean_temp" = "olivedrab3",
"med_temp" = "seagreen3",
"max_temp" = "tomato",
"min_temp" = "dodgerblue",
"range_temp" = "goldenrod3",
"var_temp" = "darkgoldenrod1"
)) +
scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
ggtitle("Daily Values") +
labs(y = "Temperature (°C)",
x = "") +
theme_bw(base_size = 20) +
theme(panel.grid = element_blank(),
axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
## Defining the function to get predictor values for periods of different lengths
get_predictors = function(daily_values, raw_temp, n_days){
prefix = str_replace_all(xfun::numbers_to_words(n_days), pattern = " ", replacement = "-")
mean_values = daily_values %>%
ungroup() %>%
mutate(mean_max = slide_vec(.x = max_temp, .f = mean, .before = n_days, .complete = T),
mean_min = slide_vec(.x = min_temp, .f = mean, .before = n_days, .complete = T),
mean_range = slide_vec(.x = range_temp, .f = mean, .before = n_days, .complete = T)) %>%
select(date, mean_max, mean_min, mean_range) %>%
rename_with( ~ paste(prefix, "day", .x, sep = "_"), .cols = c(-date))
period_values = raw_temp %>%
mutate(mean = slide_index_mean(temp, i = date, before = days(n_days),
na_rm = T),
max = slide_index_max(temp, i = date, before = days(n_days),
na_rm = T),
min = slide_index_min(temp, i = date, before = days(n_days),
na_rm = T),
med = slide_index_dbl(temp, .i = date, .before = days(n_days),
na_rm = T, .f = median),
var = slide_index_dbl(temp, .i = date, .before = days(n_days),
.f = var),
range = max - min) %>%
select(-temp) %>%
distinct() %>%
rename_with( ~ paste(prefix, "day", .x, sep = "_"), .cols = c(-date))%>%
inner_join(mean_values, by = c("date")) %>%
drop_na()
return(period_values)
}
### Pulling predictors and measuring correlations for much finer timescales; 1-56 days
words_to_numbers <- function(s) {
s <- stringr::str_to_lower(s)
for (i in 0:56)
s <- stringr::str_replace_all(s, words(i), as.character(i))
s
}
num_colls = full_data %>%
filter(sex == "female") %>%
select(collection_date, sp_name) %>%
distinct() %>%
count(sp_name) %>%
filter(n >= 5)
corr_vals = data.frame()
dur_vals = c(2, 3, 4, 5, 10, 15, 20, 25, 30)
for(i in dur_vals){
duration_temps = get_predictors(daily_values = daily_temp_data,
raw_temp = temp_data,
n_days = i) %>%
filter(date %in% as_date(unique(full_data$collection_date)))
corr_data = full_data %>%
filter(sp_name %in% num_colls$sp_name) %>%
filter(sex == "female") %>%
mutate(collection_date = as.Date(collection_date)) %>%
inner_join(duration_temps, join_by(collection_date == date)) %>%
pivot_longer(cols = c(collection_temp, contains("day_")),
values_to = "value",
names_to = "predictor") %>%
group_by(sp_name, predictor) %>%
summarise(correlation = cor.test(ctmax, value)$estimate,
p.value = cor.test(ctmax, value)$p.value,
ci_low = cor.test(ctmax, value)$conf.int[1],
ci_high = cor.test(ctmax, value)$conf.int[2]) %>%
filter(predictor != "collection_temp") %>%
mutate(sig = ifelse(p.value <0.05, "Sig.", "Non Sig.")) %>%
separate(predictor, "_day_", into = c(NA, "parameter")) %>%
mutate(duration = i)
corr_vals = bind_rows(corr_vals, corr_data)
}
corr_vals = corr_vals %>%
mutate(duration = as.numeric(duration))
ggplot(corr_vals, aes(x = duration, y = correlation, colour = sp_name)) +
facet_wrap(.~parameter) +
geom_hline(yintercept = 0) +
geom_line(linewidth = 1) +
theme_matt_facets()

# ## Getting predictor variables for different periods
#
# ### Short (three days)
# three_day_temps = get_predictors(daily_values = daily_temp_data,
# raw_temp = temp_data,
# n_days = 3)
#
# ### ONE WEEK
week_temps = get_predictors(daily_values = daily_temp_data,
raw_temp = temp_data,
n_days = 7)
#
# week_plot = week_temps %>%
# pivot_longer(cols = c(-date),
# names_to = "parameter",
# values_to = "temp") %>%
# filter(parameter %in% c("seven_day_mean",
# "seven_day_med",
# "seven_day_max",
# "seven_day_min",
# "seven_day_var",
# "seven_day_range")) %>%
# mutate(parameter = paste(word(parameter, start = 3, sep = fixed("_")), "_temp", sep = "")) %>%
# ggplot(aes(x = date, y = temp, colour = parameter)) +
# geom_line(linewidth = 1) +
# scale_colour_manual(values = c(
# "mean_temp" = "olivedrab3",
# "med_temp" = "seagreen3",
# "max_temp" = "tomato",
# "min_temp" = "dodgerblue",
# "range_temp" = "goldenrod3",
# "var_temp" = "darkgoldenrod1"
# )) +
# scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
# ggtitle("One Week") +
# labs(y = "Temperature (°C)",
# x = "") +
# theme_bw(base_size = 20) +
# theme(panel.grid = element_blank(),
# axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
#
#
# ### TWO WEEKS
# two_week_temps = get_predictors(daily_values = daily_temp_data,
# raw_temp = temp_data,
# n_days = 14)
#
# two_week_plot = two_week_temps %>%
# pivot_longer(cols = c(-date),
# names_to = "parameter",
# values_to = "temp") %>%
# filter(parameter %in% c("fourteen_day_mean",
# "fourteen_day_med",
# "fourteen_day_max",
# "fourteen_day_min",
# "fourteen_day_var",
# "fourteen_day_range")) %>%
# mutate(parameter = paste(word(parameter, start = 3, sep = fixed("_")), "_temp", sep = "")) %>%
# ggplot(aes(x = date, y = temp, colour = parameter)) +
# geom_line(linewidth = 1) +
# scale_colour_manual(values = c(
# "mean_temp" = "olivedrab3",
# "med_temp" = "seagreen3",
# "max_temp" = "tomato",
# "min_temp" = "dodgerblue",
# "range_temp" = "goldenrod3",
# "var_temp" = "darkgoldenrod1"
# )) +
# scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
# ggtitle("Two Weeks") +
# labs(y = "Temperature (°C)",
# x = "") +
# theme_bw(base_size = 20) +
# theme(panel.grid = element_blank(),
# axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
#
#
# ### FOUR WEEKS
four_week_temps = get_predictors(daily_values = daily_temp_data,
raw_temp = temp_data,
n_days = 28)
#
# four_week_plot = four_week_temps %>%
# pivot_longer(cols = c(-date),
# names_to = "parameter",
# values_to = "temp") %>%
# filter(parameter %in% c("twenty-eight_day_mean",
# "twenty-eight_day_med",
# "twenty-eight_day_max",
# "twenty-eight_day_min",
# "twenty-eight_day_var",
# "twenty-eight_day_range")) %>%
# mutate(parameter = paste(word(parameter, start = 3, sep = fixed("_")), "_temp", sep = "")) %>%
# ggplot(aes(x = date, y = temp, colour = parameter)) +
# geom_line(linewidth = 1) +
# scale_colour_manual(values = c(
# "mean_temp" = "olivedrab3",
# "med_temp" = "seagreen3",
# "max_temp" = "tomato",
# "min_temp" = "dodgerblue",
# "range_temp" = "goldenrod3",
# "var_temp" = "darkgoldenrod1"
# )) +
# scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
# ggtitle("Four Weeks") +
# labs(y = "Temperature (°C)",
# x = "") +
# theme_bw(base_size = 20) +
# theme(panel.grid = element_blank(),
# axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
#
#
# ### EIGHT WEEKS
# eight_week_temps = get_predictors(daily_values = daily_temp_data,
# raw_temp = temp_data,
# n_days = 56)
#
# eight_week_plot = eight_week_temps %>%
# pivot_longer(cols = c(-date),
# names_to = "parameter",
# values_to = "temp") %>%
# filter(parameter %in% c("fifty-six_day_mean",
# "fifty-six_day_med",
# "fifty-six_day_max",
# "fifty-six_day_min",
# "fifty-six_day_var",
# "fifty-six_day_range")) %>%
# mutate(parameter = paste(word(parameter, start = 3, sep = fixed("_")), "_temp", sep = "")) %>%
# ggplot(aes(x = date, y = temp, colour = parameter)) +
# geom_line(linewidth = 1) +
# scale_colour_manual(values = c(
# "mean_temp" = "olivedrab3",
# "med_temp" = "seagreen3",
# "max_temp" = "tomato",
# "min_temp" = "dodgerblue",
# "range_temp" = "goldenrod3",
# "var_temp" = "darkgoldenrod1"
# )) +
# scale_x_continuous(breaks = as.Date(c("2023-01-01", "2023-04-01", "2023-07-01"))) +
# ggtitle("Eight Weeks") +
# labs(y = "Temperature (°C)",
# x = "") +
# theme_bw(base_size = 20) +
# theme(panel.grid = element_blank(),
# axis.text.x = element_text(angle = 270, hjust = 0, vjust = 0.5))
#
# ggarrange(daily_plot, week_plot, two_week_plot, four_week_plot, eight_week_plot,
# common.legend = T, nrow = 1, legend = "bottom")
The different time periods examined by this climate data highlights
that the relationship between minimum and maximum temperatures changes
based on the window examined. For example, minimum and maximum
temperatures experienced over weekly intervals are closely linked,
whereas there is a distinct seasonal cycle in the relationship between
minimum and maximum temperatures experienced over periods of four
weeks.
one_week_doy_data = week_temps %>%
mutate(doy = yday(date))
one_week_temp_circle = ggplot(one_week_doy_data, aes(x = seven_day_mean_max, y = seven_day_mean_min, colour = doy)) +
geom_point() +
scale_colour_gradient(
high = "coral2",
low = "dodgerblue4") +
labs(x = "Max. Temp. (°C)",
y = "Min. Temp. (°C)") +
labs(x = "Max. Temp. (°C)",
y = "Min. Temp. (°C)") +
ggtitle("One Week") +
theme_matt()
four_week_doy_data = four_week_temps %>%
mutate(doy = yday(date))
four_week_temp_circle = ggplot(four_week_doy_data, aes(x = `twenty-eight_day_max`, y = `twenty-eight_day_min`, colour = doy)) +
geom_point() +
scale_colour_gradient(
high = "coral2",
low = "dodgerblue4") +
labs(x = "Max. Temp. (°C)",
y = "Min. Temp. (°C)") +
ggtitle("Four Week") +
theme_matt()
ggarrange(one_week_temp_circle, four_week_temp_circle,
common.legend = T, legend = "bottom")

## Daily values for the period examined by dataset
collection_conditions = temp_data %>%
ungroup() %>%
group_by(date) %>%
summarise(mean_temp = mean(temp),
med_temp = median(temp),
var_temp = var(temp),
min_temp = min(temp),
max_temp = max(temp)) %>%
mutate("range_temp" = max_temp - min_temp,
date = as.Date(date)) %>%
ungroup() %>%
filter(date >= (min(as.Date(full_data$collection_date)) - 7))
## Mean female thermal limits for each species, grouped by collection
species_summaries = full_data %>%
#filter(sex == "female") %>%
group_by(sp_name, collection_date, collection_temp) %>%
summarise("mean_ctmax" = mean(ctmax),
"sample_size" = n(),
"ctmax_st_err" = (sd(ctmax) / sqrt(sample_size)),
"ctmax_var" = var(ctmax),
"mean_size" = mean(size),
"size_st_err" = (sd(size) / sqrt(sample_size)),
"size_var" = var(size)) %>%
ungroup() %>%
complete(sp_name, collection_date) %>%
arrange(desc(sample_size))
adult_summaries = full_data %>%
filter(sex == "female") %>%
group_by(sp_name, collection_date, collection_temp) %>%
summarise("mean_ctmax" = mean(ctmax),
"sample_size" = n(),
"ctmax_st_err" = (sd(ctmax) / sqrt(sample_size)),
"ctmax_var" = var(ctmax),
"mean_size" = mean(size),
"size_st_err" = (sd(size) / sqrt(sample_size)),
"size_var" = var(size)) %>%
ungroup() %>%
complete(sp_name, collection_date) %>%
arrange(desc(sample_size))
ggplot() +
geom_vline(data = unique(select(full_data, collection_date)),
aes(xintercept = as.Date(collection_date)),
colour = "grey90",
linewidth = 1) +
geom_line(data = collection_conditions,
aes(x = as.Date(date), y = mean_temp),
colour = "black",
linewidth = 2) +
# geom_errorbar(data = species_summaries,
# aes(x = as.Date(collection_date),
# ymin = mean_ctmax - ctmax_st_err, ymax = mean_ctmax + ctmax_st_err,
# colour = sp_name),
# position = position_dodge(width = 1),
# width = 5, linewidth = 1) +
geom_point(data = species_summaries,
aes(x = as.Date(collection_date), y = mean_ctmax, colour = sp_name, size = sample_size)) +
scale_colour_manual(values = species_cols) +
labs(x = "Date",
y = "Temperature (°C)",
colour = "Species",
size = "Sample Size") +
theme_matt() +
theme(legend.position = "right")

size_timeseries = ggplot() +
geom_vline(data = unique(select(full_data, collection_date)),
aes(xintercept = as.Date(collection_date)),
colour = "grey90",
linewidth = 1) +
geom_line(data = collection_conditions,
aes(x = as.Date(date), y = mean_temp),
colour = "black",
linewidth = 2) +
# geom_errorbar(data = species_summaries,
# aes(x = as.Date(collection_date),
# ymin = mean_ctmax - ctmax_st_err, ymax = mean_ctmax + ctmax_st_err,
# colour = sp_name),
# position = position_dodge(width = 1),
# width = 5, linewidth = 1) +
geom_point(data = species_summaries,
aes(x = as.Date(collection_date), y = mean_size * 40, colour = sp_name),
position = position_dodge(width = 1),
size = 4) +
scale_colour_manual(values = species_cols) +
scale_y_continuous(
name = "Temperature", # Features of the first axis
sec.axis = sec_axis(~./40, name="Prosome Length (mm)"), # Add a second axis and specify its features
breaks = c(0,5,10,15,20,25,30)
) +
labs(x = "Date",
y = "Temperature (°C)",
colour = "Species") +
theme_matt() +
theme(legend.position = "right")
#ggarrange(ctmax_timeseries, size_timeseries, common.legend = T, legend = "bottom")
# ## Combine data, then pull out values for each collection date
# date_list = as.Date(unique(full_data$collection_date))
#
# temp_predictors = daily_temp_data %>%
# full_join(day_prior_temp_data, by = c("date")) %>%
# full_join(three_day_temps, by = c("date")) %>%
# full_join(week_temps, by = c("date")) %>%
# full_join(two_week_temps, by = c("date")) %>%
# full_join(four_week_temps, by = c("date")) %>%
# full_join(eight_week_temps, by = c("date")) %>%
# filter(date %in% date_list)
A set of predictors variables were assembled from the continuous
temperature data set based on conditions during the day of collection,
the week before collections, and the preceding two, four, and eight week
periods. This is a preliminary analysis for now. Shown here are the top
three factors. Species with no significant predictor or limited
collection date distributions were excluded.
#
# corr_vals = full_data %>%
# filter(sp_name %in% num_colls$sp_name) %>%
# filter(sex == "female") %>%
# mutate(collection_date = as.Date(collection_date)) %>%
# full_join(temp_predictors, join_by(collection_date == date)) %>%
# pivot_longer(cols = c(collection_temp, mean_temp:tail(names(.), 1)),
# values_to = "value",
# names_to = "predictor") %>%
# group_by(sp_name, predictor) %>%
# summarise(correlation = cor.test(ctmax, value)$estimate,
# p.value = cor.test(ctmax, value)$p.value,
# ci_low = cor.test(ctmax, value)$conf.int[1],
# ci_high = cor.test(ctmax, value)$conf.int[2]) %>%
# mutate(sig = ifelse(p.value <0.05, "Sig.", "Non Sig."))
corr_vals %>%
filter(sig == "Sig.") %>%
drop_na(correlation) %>%
group_by(sp_name) %>%
arrange(desc(correlation)) %>%
slice_head(n = 3) %>%
select("Species" = sp_name, "Predictor" = parameter, "Duration" = duration, "Correlation" = correlation, "P-Value" = p.value) %>%
knitr::kable(align = "c")
| Epischura lacustris |
max |
2 |
0.7250065 |
1.28e-05 |
| Epischura lacustris |
med |
5 |
0.7245285 |
1.30e-05 |
| Epischura lacustris |
mean |
3 |
0.7218370 |
1.45e-05 |
| Leptodiaptomus minutus |
mean_max |
4 |
0.5992436 |
0.00e+00 |
| Leptodiaptomus minutus |
max |
10 |
0.5987724 |
0.00e+00 |
| Leptodiaptomus minutus |
mean_max |
5 |
0.5984147 |
0.00e+00 |
| Skistodiaptomus oregonensis |
max |
2 |
0.5426235 |
0.00e+00 |
| Skistodiaptomus oregonensis |
mean_max |
2 |
0.5391528 |
0.00e+00 |
| Skistodiaptomus oregonensis |
mean_max |
3 |
0.5358973 |
0.00e+00 |
Trait Variation
ctmax_plot = full_data %>%
mutate( #sp_name = str_replace(sp_name, pattern = " ",
# replacement = "\n"),
sp_name = fct_reorder(sp_name, ctmax, mean)) %>%
ggplot(aes(y = sp_name, x = ctmax)) +
geom_point(aes(colour= sp_name_sub),
position = position_dodge(width = 0.3),
size = 4) +
scale_colour_manual(values = species_cols) +
xlab(NULL) +
labs(y = "",
x = "CTmax (°C)",
colour = "Group") +
theme_matt() +
theme(legend.position = "none")
size_plot = full_data %>%
mutate(sp_name = fct_reorder(sp_name, ctmax, mean)) %>%
ggplot(aes(y = sp_name, x = size)) +
geom_point(aes(colour= sp_name_sub),
position = position_dodge(width = 0.3),
size = 4) +
scale_colour_manual(values = species_cols) +
labs(x = "Prosome Length (mm)",
y = "",
colour = "Group") +
guides(color = guide_legend(ncol = 1)) +
theme_matt(base_size = ) +
theme(legend.position = "right",
axis.text.y = element_blank(),
plot.margin = margin(0, 0, 0, 0,"cm"))
trait_plot = ctmax_plot + size_plot
trait_plot

full_data %>%
drop_na(fecundity) %>%
ggplot(aes(x = fecundity, fill = sp_name_sub)) +
facet_wrap(.~sp_name_sub, ncol = 1) +
geom_histogram(binwidth = 2) +
scale_fill_manual(values = species_cols) +
labs(x = "Fecundity (# Eggs)") +
theme_matt_facets() +
theme(legend.position = "none")

Variation with temperature
ctmax_temp = ggplot(full_data, aes(x = collection_temp, y = ctmax, colour = sp_name)) +
geom_smooth(method = "lm", linewidth = 3) +
geom_point(size = 3) +
labs(x = "Collection Temperature (°C)",
y = "CTmax (°C)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")
size_temp = ggplot(filter(full_data, sex != "juvenile"), aes(x = collection_temp, y = size, colour = sp_name)) +
geom_smooth(method = "lm", linewidth = 3) +
geom_point(size = 3) +
labs(x = "Collection Temperature (°C)",
y = "Length (mm)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")
wt_temp = ggplot(full_data, aes(x = collection_temp, y = warming_tol, colour = sp_name)) +
geom_smooth(method = "lm", linewidth = 3) +
geom_point(size = 3) +
labs(x = "Collection Temperature (°C)",
y = "Warming Tolerance (°C)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")
eggs_temp = ggplot(full_data, aes(x = collection_temp, y = fecundity, colour = sp_name)) +
geom_smooth(method = "lm", linewidth = 3) +
geom_point(size = 3) +
labs(x = "Collection Temperature (°C)",
y = "Fecundity (# Eggs)",
colour = "Species") +
scale_colour_manual(values = species_cols) +
theme_matt() +
theme(legend.position = "right")
ggarrange(ctmax_temp, size_temp, wt_temp, eggs_temp,
common.legend = T, legend = "right")

# adult_data = full_data %>%
# filter(sex == "female")
model_data = full_data %>%
drop_na(size, ctmax)
ctmax_temp.model = lm(data = model_data, ctmax ~ collection_temp * sp_name + sex)
size_temp.model = lm(data = model_data, size ~ collection_temp * sp_name + sex)
knitr::kable(car::Anova(ctmax_temp.model))
| collection_temp |
298.388536 |
1 |
235.051589 |
0.0000000 |
| sp_name |
2027.053911 |
6 |
266.130780 |
0.0000000 |
| sex |
7.799049 |
2 |
3.071798 |
0.0472793 |
| collection_temp:sp_name |
3.366110 |
4 |
0.662902 |
0.6180352 |
| Residuals |
590.298792 |
465 |
NA |
NA |
ctmax_resids = cbind(model_data, "resids" = ctmax_temp.model$residuals, "size_resids" = size_temp.model$residuals)
ggplot(ctmax_resids, aes(x = days_in_lab, y = resids, colour = sp_name)) +
facet_wrap(sp_name~.) +
geom_point(size = 4, alpha = 0.5) +
geom_smooth(method = "lm", se = F, linewidth = 2) +
scale_x_continuous(breaks = c(0:5)) +
labs(x = "Days in lab",
y = "CTmax Residuals") +
scale_colour_manual(values = species_cols) +
theme_matt_facets() +
theme(legend.position = "none")

Given the long generation times of these copepods, decreases in trait
variance may indicate selection over the seasonal cycle. Shown below are
the variance in observed CTmax and size, plotted against collection
date. Variance decreases in Skistodiaptomus, but this pattern
is driven by a single collection with high variance early in the year.
Size variance increases slightly in Skistodiaptomus. Variance
in both CTmax and size is fairly constant in Leptodiaptomus
minutus, the only other species collected across the entire set of
samples thus far.
ggplot(drop_na(adult_summaries, ctmax_var), aes(x = as.Date(collection_date), y = ctmax_var, colour = sp_name)) +
facet_wrap(sp_name~., scales = "free_y") +
geom_point(size = 2) +
geom_smooth(method = "lm", se = F) +
labs(x = "Collection Temp. (°C)",
y = "CTmax Variance") +
scale_colour_manual(values = species_cols) +
theme_matt_facets() +
theme(legend.position = "none")

ggplot(drop_na(adult_summaries, size_var), aes(x = as.Date(collection_date), y = size_var, colour = sp_name)) +
facet_wrap(sp_name~.) +
geom_point(size = 2) +
geom_smooth(method = "lm", se = F) +
labs(x = "Collection Temp. (°C)",
y = "Size Variance") +
scale_colour_manual(values = species_cols) +
theme_matt_facets() +
theme(legend.position = "none")

LS0tCnRpdGxlOiBTZWFzb25hbGl0eSBpbiBMYWtlIENoYW1wbGFpbiBDb3BlcG9kIFRoZXJtYWwgTGltaXRzCmRhdGU6ICJgciBTeXMuRGF0ZSgpYCIKb3V0cHV0OiAKICBodG1sX2RvY3VtZW50OgogICAgICAgICAgY29kZV9mb2xkaW5nOiBoaWRlCiAgICAgICAgICBjb2RlX2Rvd25sb2FkOiB0cnVlCiAgICAgICAgICB0b2M6IHRydWUKICAgICAgICAgIHRvY19mbG9hdDogdHJ1ZQogIGdpdGh1Yl9kb2N1bWVudDoKICAgICAgICAgIGh0bWxfcHJldmlldzogZmFsc2UKICAgICAgICAgIHRvYzogdHJ1ZQogICAgICAgICAgdG9jX2RlcHRoOiAzCi0tLQoKYGBge3IgdG8tZG99CiMjIyBUbyBEbyAKCiMgQWN0dWFsIHN0YXRpc3RpY3MgZm9yIHJlbGF0aW9uc2hpcHMgYmV0d2VlbiB0ZW1wZXJhdHVyZSBhbmQgQ1RtYXgsIHNpemUsIGFuZCBmZWN1bmRpdHkKIyBQdWxsIHJlc2lkdWFscyBmcm9tIENUbWF4IH4gdGVtcGVyYXR1cmUgbW9kZWwsIGFuZCBleGFtaW5lIHRoZSBjaGFuZ2Ugb3ZlciB0aW1lIGluIGxhYiBhbmQgdGhlIHJlbGF0aW9uc2hpcCB3aXRoIGZlY3VuZGl0eQoKYGBgCgoKYGBge3Igc2V0dXAsIGluY2x1ZGU9VCwgbWVzc2FnZSA9IEYsIHdhcm5pbmcgPSBGLCBlY2hvID0gRn0Ka25pdHI6Om9wdHNfY2h1bmskc2V0KAogIGVjaG8gPSBrbml0cjo6aXNfaHRtbF9vdXRwdXQoKSwKICBmaWcuYWxpZ24gPSAiY2VudGVyIiwKICBmaWcucGF0aCA9ICIuLi9GaWd1cmVzL21hcmtkb3duLyIsCiAgZGV2ID0gYygicG5nIiwgInBkZiIpLAogIG1lc3NhZ2UgPSBGQUxTRSwKICB3YXJuaW5nID0gRkFMU0UsCiAgY29sbGFwc2UgPSBUCikKCnRoZW1lX21hdHQgPSBmdW5jdGlvbihiYXNlX3NpemUgPSAxOCwKICAgICAgICAgICAgICAgICAgICAgIGRhcmtfdGV4dCA9ICJncmV5MjAiKXsKICBtaWRfdGV4dCA8LSAgbW9ub2Nocm9tZVI6OmdlbmVyYXRlX3BhbGV0dGUoZGFya190ZXh0LCAiZ29fbGlnaHRlciIsIG5fY29sb3VycyA9IDUpWzJdCiAgbGlnaHRfdGV4dCA8LSAgbW9ub2Nocm9tZVI6OmdlbmVyYXRlX3BhbGV0dGUoZGFya190ZXh0LCAiZ29fbGlnaHRlciIsIG5fY29sb3VycyA9IDUpWzNdCiAgCiAgZ2dwdWJyOjp0aGVtZV9wdWJyKGJhc2VfZmFtaWx5PSJzYW5zIikgJStyZXBsYWNlJSAKICAgIHRoZW1lKAogICAgICBwYW5lbC5iYWNrZ3JvdW5kICA9IGVsZW1lbnRfcmVjdChmaWxsPSJ0cmFuc3BhcmVudCIsIGNvbG91cj1OQSksIAogICAgICBwbG90LmJhY2tncm91bmQgPSBlbGVtZW50X3JlY3QoZmlsbD0idHJhbnNwYXJlbnQiLCBjb2xvdXI9TkEpLCAKICAgICAgbGVnZW5kLmJhY2tncm91bmQgPSBlbGVtZW50X3JlY3QoZmlsbD0idHJhbnNwYXJlbnQiLCBjb2xvdXI9TkEpLAogICAgICBsZWdlbmQua2V5ID0gZWxlbWVudF9yZWN0KGZpbGw9InRyYW5zcGFyZW50IiwgY29sb3VyPU5BKSwKICAgICAgdGV4dCA9IGVsZW1lbnRfdGV4dChjb2xvdXIgPSBtaWRfdGV4dCwgbGluZWhlaWdodCA9IDEuMSksCiAgICAgIHRpdGxlID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUgKiAxLjUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbG91ciA9IGRhcmtfdGV4dCksCiAgICAgIGF4aXMudGV4dCA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sb3VyID0gbWlkX3RleHQpLAogICAgICBheGlzLnRpdGxlLnggPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSAqIDEuMiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1hcmdpbiA9IHVuaXQoYygzLCAwLCAwLCAwKSwgIm1tIikpLAogICAgICBheGlzLnRpdGxlLnkgPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSAqIDEuMiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG1hcmdpbiA9IHVuaXQoYygwLCA1LCAwLCAwKSwgIm1tIiksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYW5nbGUgPSA5MCksCiAgICAgIGxlZ2VuZC50ZXh0ID0gZWxlbWVudF90ZXh0KHNpemU9YmFzZV9zaXplICogMC45KSwKICAgICAgbGVnZW5kLnRpdGxlID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUgKiAwLjksIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmFjZSA9ICJib2xkIiksCiAgICAgIHBsb3QubWFyZ2luID0gbWFyZ2luKDAuMjUsIDAuMjUsIDAuMjUsIDAuMjUsImNtIikKICAgICkKfQoKdGhlbWVfbWF0dF9mYWNldHMgPSBmdW5jdGlvbihiYXNlX3NpemUgPSAxOCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkYXJrX3RleHQgPSAiZ3JleTIwIil7CiAgbWlkX3RleHQgPC0gIG1vbm9jaHJvbWVSOjpnZW5lcmF0ZV9wYWxldHRlKGRhcmtfdGV4dCwgImdvX2xpZ2h0ZXIiLCBuX2NvbG91cnMgPSA1KVsyXQogIGxpZ2h0X3RleHQgPC0gIG1vbm9jaHJvbWVSOjpnZW5lcmF0ZV9wYWxldHRlKGRhcmtfdGV4dCwgImdvX2xpZ2h0ZXIiLCBuX2NvbG91cnMgPSA1KVszXQogIAogIHRoZW1lX2J3KGJhc2VfZmFtaWx5PSJzYW5zIikgJStyZXBsYWNlJSAKICAgIHRoZW1lKAogICAgICBwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAogICAgICBwYW5lbC5iYWNrZ3JvdW5kICA9IGVsZW1lbnRfcmVjdChmaWxsPSJ0cmFuc3BhcmVudCIsIGNvbG91cj1OQSksIAogICAgICBwbG90LmJhY2tncm91bmQgPSBlbGVtZW50X3JlY3QoZmlsbD0idHJhbnNwYXJlbnQiLCBjb2xvdXI9TkEpLCAKICAgICAgbGVnZW5kLmJhY2tncm91bmQgPSBlbGVtZW50X3JlY3QoZmlsbD0idHJhbnNwYXJlbnQiLCBjb2xvdXI9TkEpLAogICAgICBsZWdlbmQua2V5ID0gZWxlbWVudF9yZWN0KGZpbGw9InRyYW5zcGFyZW50IiwgY29sb3VyPU5BKSwKICAgICAgdGV4dCA9IGVsZW1lbnRfdGV4dChjb2xvdXIgPSBtaWRfdGV4dCwgbGluZWhlaWdodCA9IDEuMSksCiAgICAgIHN0cmlwLnRleHQueCA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplKSwKICAgICAgdGl0bGUgPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSAqIDEuNSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgY29sb3VyID0gZGFya190ZXh0KSwKICAgICAgYXhpcy50ZXh0ID0gZWxlbWVudF90ZXh0KHNpemUgPSBiYXNlX3NpemUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb2xvdXIgPSBtaWRfdGV4dCksCiAgICAgIGF4aXMudGl0bGUueCA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplICogMS4yLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbWFyZ2luID0gdW5pdChjKDMsIDAsIDAsIDApLCAibW0iKSksCiAgICAgIGF4aXMudGl0bGUueSA9IGVsZW1lbnRfdGV4dChzaXplID0gYmFzZV9zaXplICogMS4yLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbWFyZ2luID0gdW5pdChjKDAsIDUsIDAsIDApLCAibW0iKSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhbmdsZSA9IDkwKSwKICAgICAgbGVnZW5kLnRleHQgPSBlbGVtZW50X3RleHQoc2l6ZT1iYXNlX3NpemUgKiAwLjkpLAogICAgICBsZWdlbmQudGl0bGUgPSBlbGVtZW50X3RleHQoc2l6ZSA9IGJhc2Vfc2l6ZSAqIDAuOSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmYWNlID0gImJvbGQiKSwKICAgICAgcGxvdC5tYXJnaW4gPSBtYXJnaW4oMC4yNSwgMC4yNSwgMC4yNSwgMC4yNSwiY20iKQogICAgKQp9CgpzcGVjaWVzX2NvbHMgPSBjKCJMZXB0b2RpYXB0b211cyBtaW51dHVzIiA9ICIjZmZkMDI5IiwKICAgICAgICAgICAgICAgICAiTGVwdG9kaWFwdG9tdXMgbWludXR1cyBqdXZlbmlsZSIgPSAiI2UzZDhhZiIsCiAgICAgICAgICAgICAgICAgIkxlcHRvZGlhcHRvbXVzIG1pbnV0dXMgbWFsZSIgPSAiI2ZmZTg5NiIsCiAgICAgICAgICAgICAgICAgIkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMiID0gIiNEODZGMjkiLAogICAgICAgICAgICAgICAgICJMZXB0b2RpYXB0b211cyBzaWNpbGlzIG1hbGUiID0gIiNFMjhDMDAiLAogICAgICAgICAgICAgICAgICJTa2lzdG9kaWFwdG9tdXMgb3JlZ29uZW5zaXMiID0gIiNDNUMzNUEiLAogICAgICAgICAgICAgICAgICJTa2lzdG9kaWFwdG9tdXMgb3JlZ29uZW5zaXMgbWFsZSIgPSAiI2U2ZTZhYSIsIAogICAgICAgICAgICAgICAgICJFcGlzY2h1cmEgbGFjdXN0cmlzIGp1dmVuaWxlIiA9ICJwbHVtMSIsIAogICAgICAgICAgICAgICAgICJFcGlzY2h1cmEgbGFjdXN0cmlzIG1hbGUiID0gInBsdW0zIiwgCiAgICAgICAgICAgICAgICAgIkVwaXNjaHVyYSBsYWN1c3RyaXMiID0gInBsdW00IiwgCiAgICAgICAgICAgICAgICAgIkxpbW5vY2FsYW51cyBtYWNydXJ1cyIgPSAic2t5Ymx1ZTQiLCAKICAgICAgICAgICAgICAgICAiTGltbm9jYWxhbnVzIG1hY3J1cnVzIG1hbGUiID0gInNreWJsdWUzIiwgCiAgICAgICAgICAgICAgICAgIkxpbW5vY2FsYW51cyBtYWNydXJ1cyBqdXZlbmlsZSIgPSAic2t5Ymx1ZSIsIAogICAgICAgICAgICAgICAgICJTZW5lY2VsbGEgY2FsYW5vaWRlcyIgPSAiZGFya3NlYWdyZWVuMyIsCiAgICAgICAgICAgICAgICAgIkxlcHRvZG9yYSBraW5kdGkgbWFsZSIgPSAibGlnaHRibHVlMyIsCiAgICAgICAgICAgICAgICAgIkxlcHRvZG9yYSBraW5kdGkiID0gImxpZ2h0Ymx1ZTQiLAogICAgICAgICAgICAgICAgICJMZXB0b2RvcmEga2luZHRpIGp1dmVuaWxlIiA9ICJsaWdodGJsdWUiLAogICAgICAgICAgICAgICAgICJPc3BocmFudGljdW0gbGFicm9uZWN0dW0iID0gImxpZ2h0Y29yYWwiKQpgYGAKCiMjIENvcGVwb2QgQ29sbGVjdGlvbgoKQ29wZXBvZHMgd2VyZSBjb2xsZWN0ZWQgYXQgYXBwcm94aW1hdGVseSB3ZWVrbHkgaW50ZXJ2YWxzIGZyb20gTGFrZSBDaGFtcGxhaW4gKEJ1cmxpbmd0b24gRmlzaGluZyBQaWVyKS4gUGxhbmt0b24gd2FzIGNvbGxlY3RlZCBmcm9tIHRoZSB0b3AgMyBtZXRlcnMgdXNpbmcgYSAyNTAgdW0gbWVzaCBuZXQuIENvcGVwb2RzIGZyb20gYHIgbGVuZ3RoKHVuaXF1ZShmdWxsX2RhdGEkY29sbGVjdGlvbl9kYXRlKSlgIGNvbGxlY3Rpb25zIHdlcmUgdXNlZCB0byBtYWtlIGEgdG90YWwgb2YgYHIgZGltKGZ1bGxfZGF0YSlbMV1gIHRoZXJtYWwgbGltaXQgbWVhc3VyZW1lbnRzLiBPdmVyIHRoaXMgdGltZSBwZXJpb2QsIGNvbGxlY3Rpb24gdGVtcGVyYXR1cmVzIHJhbmdlZCBmcm9tIGByIHBhc3RlKG1pbihmdWxsX2RhdGEkY29sbGVjdGlvbl90ZW1wKSwgIiB0byAiLCBtYXgoZnVsbF9kYXRhJGNvbGxlY3Rpb25fdGVtcCksIHNlcCA9ICIiKWDCsEMuIAoKIyMgVGVtcGVyYXR1cmUgVmFyaWF0aW9uIApXYXRlciB0ZW1wZXJhdHVyZXMgaW4gTGFrZSBDaGFtcGxhaW4gcmVhY2hlZCBhIG1pbmltdW0gaW4gRmVicnVhcnkuIFNhbXBsaW5nIGZvciB0aGlzIHByb2plY3QgYmVnYW4gZHVyaW5nIHRoZSBTcHJpbmcgd2FybWluZyBwZXJpb2QuIFRlbXBlcmF0dXJlIHZhcmlhYmlsaXR5IChib3RoIHJhbmdlcyBhbmQgdmFyaWFuY2UpIGluY3JlYXNlIHdpdGggdGVtcGVyYXR1cmUsIGJ1dCBhcmUgc3Ryb25nbHkgYWZmZWN0ZWQgYnkgdGhlIGxlbmd0aCBvZiB0aW1lIHBlcmlvZCBleGFtaW5lZC4gICAgCgpgYGB7ciBwdWxsaW5nLXRlbXAtZGF0YX0KIyBMYWtlIENoYW1wbGFpbiBuZWFyIEJ1cmxpbmd0b24sIFZUCnNpdGVOdW1iZXIgPSAiMDQyOTQ1MDAiCkNoYW1wbGFpbkluZm8gPSByZWFkTldJU3NpdGUoc2l0ZU51bWJlcikKcGFyYW1ldGVyQ2QgPSAiMDAwMTAiCnN0YXJ0RGF0ZSA9ICIyMDIzLTAxLTAxIgplbmREYXRlID0gIiIKI3N0YXRDZCA9IGMoIjAwMDAxIiwgIjAwMDAyIiwiMDAwMDMiLCAiMDAwMTEiKSAjIDEgLSBtYXgsIDIgLSBtaW4sIDMgPSBtZWFuCgojIENvbnN0cnVjdHMgdGhlIFVSTCBmb3IgdGhlIGRhdGEgd2FudGVkIHRoZW4gZG93bmxvYWRzIHRoZSBkYXRhCnVybCA9IGNvbnN0cnVjdE5XSVNVUkwoc2l0ZU51bWJlcnMgPSBzaXRlTnVtYmVyLCBwYXJhbWV0ZXJDZCA9IHBhcmFtZXRlckNkLCAKICAgICAgICAgICAgICAgICAgICAgICBzdGFydERhdGUgPSBzdGFydERhdGUsIGVuZERhdGUgPSBlbmREYXRlLCBzZXJ2aWNlID0gInV2IikKCnRlbXBfZGF0YSA9IGltcG9ydFdhdGVyTUwxKHVybCwgYXNEYXRlVGltZSA9IFQpICU+JSAKICBtdXRhdGUoImRhdGUiID0gYXMuRGF0ZShkYXRlVGltZSkpICU+JSAKICBzZWxlY3QoZGF0ZSwgInRlbXAiID0gWF8wMDAxMF8wMDAwMCkKCgojIyBEYWlseSB2YWx1ZXMKZGFpbHlfdGVtcF9kYXRhID0gdGVtcF9kYXRhICU+JQogIHVuZ3JvdXAoKSAlPiUgCiAgZ3JvdXBfYnkoZGF0ZSkgJT4lIAogIHN1bW1hcmlzZShtZWFuX3RlbXAgPSBtZWFuKHRlbXApLAogICAgICAgICAgICBtZWRfdGVtcCA9IG1lZGlhbih0ZW1wKSwKICAgICAgICAgICAgdmFyX3RlbXAgPSB2YXIodGVtcCksIAogICAgICAgICAgICBtaW5fdGVtcCA9IG1pbih0ZW1wKSwgCiAgICAgICAgICAgIG1heF90ZW1wID0gbWF4KHRlbXApKSAlPiUgCiAgbXV0YXRlKCJyYW5nZV90ZW1wIiA9IG1heF90ZW1wIC0gbWluX3RlbXApCgpkYXlfcHJpb3JfdGVtcF9kYXRhID0gdGVtcF9kYXRhICU+JSAKICB1bmdyb3VwKCkgJT4lIAogIGdyb3VwX2J5KGRhdGUpICU+JSAKICBzdW1tYXJpc2UobWVhbl90ZW1wID0gbWVhbih0ZW1wKSwKICAgICAgICAgICAgbWVkX3RlbXAgPSBtZWRpYW4odGVtcCksCiAgICAgICAgICAgIHZhcl90ZW1wID0gdmFyKHRlbXApLCAKICAgICAgICAgICAgbWluX3RlbXAgPSBtaW4odGVtcCksIAogICAgICAgICAgICBtYXhfdGVtcCA9IG1heCh0ZW1wKSkgJT4lIAogIG11dGF0ZShkYXRlID0gZGF0ZSArIDEpICU+JSAKICByZW5hbWVfd2l0aCguZm4gPSB+IHBhc3RlMCgicHJpb3JfZGF5XyIsIC54KSwgLmNvbHMgPSBjKC1kYXRlKSkKCmRhaWx5X3Bsb3QgPSBkYWlseV90ZW1wX2RhdGEgJT4lIAogIHBpdm90X2xvbmdlcihjb2xzID0gYygtZGF0ZSksCiAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInBhcmFtZXRlciIsIAogICAgICAgICAgICAgICB2YWx1ZXNfdG8gPSAidGVtcCIpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBkYXRlLCB5ID0gdGVtcCwgY29sb3VyID0gcGFyYW1ldGVyKSkgKyAKICBnZW9tX2xpbmUobGluZXdpZHRoID0gMSkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IGMoCiAgICAibWVhbl90ZW1wIiA9ICJvbGl2ZWRyYWIzIiwKICAgICJtZWRfdGVtcCIgPSAic2VhZ3JlZW4zIiwKICAgICJtYXhfdGVtcCIgPSAidG9tYXRvIiwgIAogICAgIm1pbl90ZW1wIiA9ICJkb2RnZXJibHVlIiwKICAgICJyYW5nZV90ZW1wIiA9ICJnb2xkZW5yb2QzIiwKICAgICJ2YXJfdGVtcCIgPSAiZGFya2dvbGRlbnJvZDEiCiAgKSkgKyAKICBzY2FsZV94X2NvbnRpbnVvdXMoYnJlYWtzID0gYXMuRGF0ZShjKCIyMDIzLTAxLTAxIiwgIjIwMjMtMDQtMDEiLCAiMjAyMy0wNy0wMSIpKSkgKyAKICBnZ3RpdGxlKCJEYWlseSBWYWx1ZXMiKSArIAogIGxhYnMoeSA9ICJUZW1wZXJhdHVyZSAowrBDKSIsCiAgICAgICB4ID0gIiIpICsgCiAgdGhlbWVfYncoYmFzZV9zaXplID0gMjApICsgCiAgdGhlbWUocGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKSwKICAgICAgICBheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDI3MCwgaGp1c3QgPSAwLCB2anVzdCA9IDAuNSkpCmBgYAoKYGBge3IgcHJlZGljdG9ycy1mdW5jdGlvbn0KIyMgRGVmaW5pbmcgdGhlIGZ1bmN0aW9uIHRvIGdldCBwcmVkaWN0b3IgdmFsdWVzIGZvciBwZXJpb2RzIG9mIGRpZmZlcmVudCBsZW5ndGhzCmdldF9wcmVkaWN0b3JzID0gZnVuY3Rpb24oZGFpbHlfdmFsdWVzLCByYXdfdGVtcCwgbl9kYXlzKXsKICBwcmVmaXggPSBzdHJfcmVwbGFjZV9hbGwoeGZ1bjo6bnVtYmVyc190b193b3JkcyhuX2RheXMpLCBwYXR0ZXJuID0gIiAiLCByZXBsYWNlbWVudCA9ICItIikKICAKICBtZWFuX3ZhbHVlcyA9IGRhaWx5X3ZhbHVlcyAlPiUgCiAgICB1bmdyb3VwKCkgJT4lIAogICAgbXV0YXRlKG1lYW5fbWF4ID0gc2xpZGVfdmVjKC54ID0gbWF4X3RlbXAsIC5mID0gbWVhbiwgLmJlZm9yZSA9IG5fZGF5cywgLmNvbXBsZXRlID0gVCksCiAgICAgICAgICAgbWVhbl9taW4gPSBzbGlkZV92ZWMoLnggPSBtaW5fdGVtcCwgLmYgPSBtZWFuLCAuYmVmb3JlID0gbl9kYXlzLCAuY29tcGxldGUgPSBUKSwKICAgICAgICAgICBtZWFuX3JhbmdlID0gc2xpZGVfdmVjKC54ID0gcmFuZ2VfdGVtcCwgLmYgPSBtZWFuLCAuYmVmb3JlID0gbl9kYXlzLCAuY29tcGxldGUgPSBUKSkgJT4lIAogICAgc2VsZWN0KGRhdGUsIG1lYW5fbWF4LCBtZWFuX21pbiwgbWVhbl9yYW5nZSkgJT4lIAogICAgcmVuYW1lX3dpdGgoIH4gcGFzdGUocHJlZml4LCAiZGF5IiwgLngsIHNlcCA9ICJfIiksIC5jb2xzID0gYygtZGF0ZSkpCiAgCiAgcGVyaW9kX3ZhbHVlcyA9IHJhd190ZW1wICU+JSAKICAgIG11dGF0ZShtZWFuID0gc2xpZGVfaW5kZXhfbWVhbih0ZW1wLCBpID0gZGF0ZSwgYmVmb3JlID0gZGF5cyhuX2RheXMpLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuYV9ybSA9IFQpLAogICAgICAgICAgIG1heCA9IHNsaWRlX2luZGV4X21heCh0ZW1wLCBpID0gZGF0ZSwgYmVmb3JlID0gZGF5cyhuX2RheXMpLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbmFfcm0gPSBUKSwKICAgICAgICAgICBtaW4gPSBzbGlkZV9pbmRleF9taW4odGVtcCwgaSA9IGRhdGUsIGJlZm9yZSA9IGRheXMobl9kYXlzKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbmFfcm0gPSBUKSwKICAgICAgICAgICBtZWQgPSBzbGlkZV9pbmRleF9kYmwodGVtcCwgLmkgPSBkYXRlLCAuYmVmb3JlID0gZGF5cyhuX2RheXMpLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbmFfcm0gPSBULCAuZiA9IG1lZGlhbiksCiAgICAgICAgICAgdmFyID0gc2xpZGVfaW5kZXhfZGJsKHRlbXAsIC5pID0gZGF0ZSwgLmJlZm9yZSA9IGRheXMobl9kYXlzKSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC5mID0gdmFyKSwKICAgICAgICAgICByYW5nZSA9IG1heCAtIG1pbikgJT4lICAKICAgIHNlbGVjdCgtdGVtcCkgJT4lICAKICAgIGRpc3RpbmN0KCkgJT4lIAogICAgcmVuYW1lX3dpdGgoIH4gcGFzdGUocHJlZml4LCAiZGF5IiwgLngsIHNlcCA9ICJfIiksIC5jb2xzID0gYygtZGF0ZSkpJT4lIAogICAgaW5uZXJfam9pbihtZWFuX3ZhbHVlcywgYnkgPSBjKCJkYXRlIikpICU+JSAgCiAgICBkcm9wX25hKCkKICAKICByZXR1cm4ocGVyaW9kX3ZhbHVlcykKfQpgYGAKCmBgYHtyfQojIyMgUHVsbGluZyBwcmVkaWN0b3JzIGFuZCBtZWFzdXJpbmcgY29ycmVsYXRpb25zIGZvciBtdWNoIGZpbmVyIHRpbWVzY2FsZXM7IDEtNTYgZGF5cwoKd29yZHNfdG9fbnVtYmVycyA8LSBmdW5jdGlvbihzKSB7CiAgcyA8LSBzdHJpbmdyOjpzdHJfdG9fbG93ZXIocykKICBmb3IgKGkgaW4gMDo1NikKICAgIHMgPC0gc3RyaW5ncjo6c3RyX3JlcGxhY2VfYWxsKHMsIHdvcmRzKGkpLCBhcy5jaGFyYWN0ZXIoaSkpCiAgcwp9CgpudW1fY29sbHMgPSBmdWxsX2RhdGEgJT4lIAogIGZpbHRlcihzZXggPT0gImZlbWFsZSIpICU+JSAKICBzZWxlY3QoY29sbGVjdGlvbl9kYXRlLCBzcF9uYW1lKSAlPiUgIAogIGRpc3RpbmN0KCkgJT4lICAKICBjb3VudChzcF9uYW1lKSAlPiUgCiAgZmlsdGVyKG4gPj0gNSkKCmNvcnJfdmFscyA9IGRhdGEuZnJhbWUoKQoKZHVyX3ZhbHMgPSBjKDIsIDMsIDQsIDUsIDEwLCAxNSwgMjAsIDI1LCAzMCkKZm9yKGkgaW4gZHVyX3ZhbHMpewogIAogIGR1cmF0aW9uX3RlbXBzID0gZ2V0X3ByZWRpY3RvcnMoZGFpbHlfdmFsdWVzID0gZGFpbHlfdGVtcF9kYXRhLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXdfdGVtcCA9IHRlbXBfZGF0YSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gaSkgJT4lIAogICAgZmlsdGVyKGRhdGUgJWluJSBhc19kYXRlKHVuaXF1ZShmdWxsX2RhdGEkY29sbGVjdGlvbl9kYXRlKSkpCiAgCiAgY29ycl9kYXRhID0gZnVsbF9kYXRhICU+JQogICAgZmlsdGVyKHNwX25hbWUgJWluJSBudW1fY29sbHMkc3BfbmFtZSkgJT4lIAogICAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAogICAgbXV0YXRlKGNvbGxlY3Rpb25fZGF0ZSA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSkgJT4lIAogICAgaW5uZXJfam9pbihkdXJhdGlvbl90ZW1wcywgam9pbl9ieShjb2xsZWN0aW9uX2RhdGUgPT0gZGF0ZSkpICU+JSAKICAgIHBpdm90X2xvbmdlcihjb2xzID0gYyhjb2xsZWN0aW9uX3RlbXAsIGNvbnRhaW5zKCJkYXlfIikpLAogICAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJ2YWx1ZSIsIAogICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInByZWRpY3RvciIpICU+JSAgCiAgICBncm91cF9ieShzcF9uYW1lLCBwcmVkaWN0b3IpICU+JSAKICAgIHN1bW1hcmlzZShjb3JyZWxhdGlvbiA9IGNvci50ZXN0KGN0bWF4LCB2YWx1ZSkkZXN0aW1hdGUsCiAgICAgICAgICAgICAgcC52YWx1ZSA9IGNvci50ZXN0KGN0bWF4LCB2YWx1ZSkkcC52YWx1ZSwKICAgICAgICAgICAgICBjaV9sb3cgPSBjb3IudGVzdChjdG1heCwgdmFsdWUpJGNvbmYuaW50WzFdLAogICAgICAgICAgICAgIGNpX2hpZ2ggPSBjb3IudGVzdChjdG1heCwgdmFsdWUpJGNvbmYuaW50WzJdKSAlPiUgCiAgICBmaWx0ZXIocHJlZGljdG9yICE9ICJjb2xsZWN0aW9uX3RlbXAiKSAlPiUgCiAgICBtdXRhdGUoc2lnID0gaWZlbHNlKHAudmFsdWUgPDAuMDUsICJTaWcuIiwgIk5vbiBTaWcuIikpICU+JSAKICAgIHNlcGFyYXRlKHByZWRpY3RvciwgIl9kYXlfIiwgaW50byA9IGMoTkEsICJwYXJhbWV0ZXIiKSkgJT4lIAogICAgbXV0YXRlKGR1cmF0aW9uID0gaSkKICAKICBjb3JyX3ZhbHMgPSBiaW5kX3Jvd3MoY29ycl92YWxzLCBjb3JyX2RhdGEpCn0KCmNvcnJfdmFscyA9IGNvcnJfdmFscyAlPiUgIAogIG11dGF0ZShkdXJhdGlvbiA9IGFzLm51bWVyaWMoZHVyYXRpb24pKQpgYGAKCmBgYHtyIGZpZy53aWR0aD0xMCwgZmlnLmhlaWdodD02fQpnZ3Bsb3QoY29ycl92YWxzLCBhZXMoeCA9IGR1cmF0aW9uLCB5ID0gY29ycmVsYXRpb24sIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGZhY2V0X3dyYXAoLn5wYXJhbWV0ZXIpICsgCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gMCkgKyAKICBnZW9tX2xpbmUobGluZXdpZHRoID0gMSkgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cygpCmBgYAoKCmBgYHtyIHByZWRpY3RvcnMtYW5kLXBsb3RzLCBmaWcud2lkdGg9MTIsIGZpZy5oZWlnaHQ9NX0KIyAjIyBHZXR0aW5nIHByZWRpY3RvciB2YXJpYWJsZXMgZm9yIGRpZmZlcmVudCBwZXJpb2RzCiMgCiMgIyMjIFNob3J0ICh0aHJlZSBkYXlzKQojIHRocmVlX2RheV90ZW1wcyA9IGdldF9wcmVkaWN0b3JzKGRhaWx5X3ZhbHVlcyA9IGRhaWx5X3RlbXBfZGF0YSwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmF3X3RlbXAgPSB0ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5fZGF5cyA9IDMpCiMgCiMgIyMjIE9ORSBXRUVLCndlZWtfdGVtcHMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBkYWlseV90ZW1wX2RhdGEsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXdfdGVtcCA9IHRlbXBfZGF0YSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5fZGF5cyA9IDcpCiMgCiMgd2Vla19wbG90ID0gd2Vla190ZW1wcyAlPiUgCiMgICBwaXZvdF9sb25nZXIoY29scyA9IGMoLWRhdGUpLAojICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInBhcmFtZXRlciIsIAojICAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJ0ZW1wIikgJT4lIAojICAgZmlsdGVyKHBhcmFtZXRlciAlaW4lIGMoInNldmVuX2RheV9tZWFuIiwKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJzZXZlbl9kYXlfbWVkIiwKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJzZXZlbl9kYXlfbWF4IiwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAic2V2ZW5fZGF5X21pbiIsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgInNldmVuX2RheV92YXIiLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgInNldmVuX2RheV9yYW5nZSIpKSAlPiUgCiMgICBtdXRhdGUocGFyYW1ldGVyID0gcGFzdGUod29yZChwYXJhbWV0ZXIsIHN0YXJ0ID0gMywgc2VwID0gZml4ZWQoIl8iKSksICJfdGVtcCIsIHNlcCA9ICIiKSkgJT4lIAojICAgZ2dwbG90KGFlcyh4ID0gZGF0ZSwgeSA9IHRlbXAsIGNvbG91ciA9IHBhcmFtZXRlcikpICsgCiMgICBnZW9tX2xpbmUobGluZXdpZHRoID0gMSkgKyAKIyAgIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gYygKIyAgICAgIm1lYW5fdGVtcCIgPSAib2xpdmVkcmFiMyIsCiMgICAgICJtZWRfdGVtcCIgPSAic2VhZ3JlZW4zIiwKIyAgICAgIm1heF90ZW1wIiA9ICJ0b21hdG8iLCAgCiMgICAgICJtaW5fdGVtcCIgPSAiZG9kZ2VyYmx1ZSIsCiMgICAgICJyYW5nZV90ZW1wIiA9ICJnb2xkZW5yb2QzIiwKIyAgICAgInZhcl90ZW1wIiA9ICJkYXJrZ29sZGVucm9kMSIKIyAgICkpICsgCiMgICBzY2FsZV94X2NvbnRpbnVvdXMoYnJlYWtzID0gYXMuRGF0ZShjKCIyMDIzLTAxLTAxIiwgIjIwMjMtMDQtMDEiLCAiMjAyMy0wNy0wMSIpKSkgKyAKIyAgIGdndGl0bGUoIk9uZSBXZWVrIikgKyAKIyAgIGxhYnMoeSA9ICJUZW1wZXJhdHVyZSAowrBDKSIsCiMgICAgICAgIHggPSAiIikgKyAKIyAgIHRoZW1lX2J3KGJhc2Vfc2l6ZSA9IDIwKSArIAojICAgdGhlbWUocGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKSwKIyAgICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKIyAKIyAKIyAjIyMgVFdPIFdFRUtTCiMgdHdvX3dlZWtfdGVtcHMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBkYWlseV90ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmF3X3RlbXAgPSB0ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gMTQpCiMgCiMgdHdvX3dlZWtfcGxvdCA9IHR3b193ZWVrX3RlbXBzICU+JSAKIyAgIHBpdm90X2xvbmdlcihjb2xzID0gYygtZGF0ZSksCiMgICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAicGFyYW1ldGVyIiwgCiMgICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInRlbXAiKSAlPiUgCiMgICBmaWx0ZXIocGFyYW1ldGVyICVpbiUgYygiZm91cnRlZW5fZGF5X21lYW4iLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZvdXJ0ZWVuX2RheV9tZWQiLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZvdXJ0ZWVuX2RheV9tYXgiLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmb3VydGVlbl9kYXlfbWluIiwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAiZm91cnRlZW5fZGF5X3ZhciIsCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAiZm91cnRlZW5fZGF5X3JhbmdlIikpICU+JSAKIyAgIG11dGF0ZShwYXJhbWV0ZXIgPSBwYXN0ZSh3b3JkKHBhcmFtZXRlciwgc3RhcnQgPSAzLCBzZXAgPSBmaXhlZCgiXyIpKSwgIl90ZW1wIiwgc2VwID0gIiIpKSAlPiUgCiMgICBnZ3Bsb3QoYWVzKHggPSBkYXRlLCB5ID0gdGVtcCwgY29sb3VyID0gcGFyYW1ldGVyKSkgKyAKIyAgIGdlb21fbGluZShsaW5ld2lkdGggPSAxKSArIAojICAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBjKAojICAgICAibWVhbl90ZW1wIiA9ICJvbGl2ZWRyYWIzIiwKIyAgICAgIm1lZF90ZW1wIiA9ICJzZWFncmVlbjMiLAojICAgICAibWF4X3RlbXAiID0gInRvbWF0byIsICAKIyAgICAgIm1pbl90ZW1wIiA9ICJkb2RnZXJibHVlIiwKIyAgICAgInJhbmdlX3RlbXAiID0gImdvbGRlbnJvZDMiLAojICAgICAidmFyX3RlbXAiID0gImRhcmtnb2xkZW5yb2QxIgojICAgKSkgKyAKIyAgIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBhcy5EYXRlKGMoIjIwMjMtMDEtMDEiLCAiMjAyMy0wNC0wMSIsICIyMDIzLTA3LTAxIikpKSArIAojICAgZ2d0aXRsZSgiVHdvIFdlZWtzIikgKyAKIyAgIGxhYnMoeSA9ICJUZW1wZXJhdHVyZSAowrBDKSIsCiMgICAgICAgIHggPSAiIikgKyAKIyAgIHRoZW1lX2J3KGJhc2Vfc2l6ZSA9IDIwKSArIAojICAgdGhlbWUocGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKSwKIyAgICAgICAgIGF4aXMudGV4dC54ID0gZWxlbWVudF90ZXh0KGFuZ2xlID0gMjcwLCBoanVzdCA9IDAsIHZqdXN0ID0gMC41KSkKIyAKIyAKIyAjIyMgRk9VUiBXRUVLUwpmb3VyX3dlZWtfdGVtcHMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBkYWlseV90ZW1wX2RhdGEsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJhd190ZW1wID0gdGVtcF9kYXRhLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBuX2RheXMgPSAyOCkKIyAKIyBmb3VyX3dlZWtfcGxvdCA9IGZvdXJfd2Vla190ZW1wcyAlPiUgCiMgICBwaXZvdF9sb25nZXIoY29scyA9IGMoLWRhdGUpLAojICAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInBhcmFtZXRlciIsIAojICAgICAgICAgICAgICAgIHZhbHVlc190byA9ICJ0ZW1wIikgJT4lIAojICAgZmlsdGVyKHBhcmFtZXRlciAlaW4lIGMoInR3ZW50eS1laWdodF9kYXlfbWVhbiIsCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAidHdlbnR5LWVpZ2h0X2RheV9tZWQiLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgInR3ZW50eS1laWdodF9kYXlfbWF4IiwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAidHdlbnR5LWVpZ2h0X2RheV9taW4iLCAKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJ0d2VudHktZWlnaHRfZGF5X3ZhciIsCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAidHdlbnR5LWVpZ2h0X2RheV9yYW5nZSIpKSAlPiUgCiMgICBtdXRhdGUocGFyYW1ldGVyID0gcGFzdGUod29yZChwYXJhbWV0ZXIsIHN0YXJ0ID0gMywgc2VwID0gZml4ZWQoIl8iKSksICJfdGVtcCIsIHNlcCA9ICIiKSkgJT4lIAojICAgZ2dwbG90KGFlcyh4ID0gZGF0ZSwgeSA9IHRlbXAsIGNvbG91ciA9IHBhcmFtZXRlcikpICsgCiMgICBnZW9tX2xpbmUobGluZXdpZHRoID0gMSkgKyAKIyAgIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gYygKIyAgICAgIm1lYW5fdGVtcCIgPSAib2xpdmVkcmFiMyIsCiMgICAgICJtZWRfdGVtcCIgPSAic2VhZ3JlZW4zIiwKIyAgICAgIm1heF90ZW1wIiA9ICJ0b21hdG8iLCAgCiMgICAgICJtaW5fdGVtcCIgPSAiZG9kZ2VyYmx1ZSIsCiMgICAgICJyYW5nZV90ZW1wIiA9ICJnb2xkZW5yb2QzIiwKIyAgICAgInZhcl90ZW1wIiA9ICJkYXJrZ29sZGVucm9kMSIKIyAgICkpICsgCiMgICBzY2FsZV94X2NvbnRpbnVvdXMoYnJlYWtzID0gYXMuRGF0ZShjKCIyMDIzLTAxLTAxIiwgIjIwMjMtMDQtMDEiLCAiMjAyMy0wNy0wMSIpKSkgKyAKIyAgIGdndGl0bGUoIkZvdXIgV2Vla3MiKSArIAojICAgbGFicyh5ID0gIlRlbXBlcmF0dXJlICjCsEMpIiwKIyAgICAgICAgeCA9ICIiKSArIAojICAgdGhlbWVfYncoYmFzZV9zaXplID0gMjApICsgCiMgICB0aGVtZShwYW5lbC5ncmlkID0gZWxlbWVudF9ibGFuaygpLAojICAgICAgICAgYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSAyNzAsIGhqdXN0ID0gMCwgdmp1c3QgPSAwLjUpKQojIAojIAojICMjIyBFSUdIVCBXRUVLUwojIGVpZ2h0X3dlZWtfdGVtcHMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBkYWlseV90ZW1wX2RhdGEsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByYXdfdGVtcCA9IHRlbXBfZGF0YSwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG5fZGF5cyA9IDU2KQojIAojIGVpZ2h0X3dlZWtfcGxvdCA9IGVpZ2h0X3dlZWtfdGVtcHMgJT4lIAojICAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKC1kYXRlKSwKIyAgICAgICAgICAgICAgICBuYW1lc190byA9ICJwYXJhbWV0ZXIiLCAKIyAgICAgICAgICAgICAgICB2YWx1ZXNfdG8gPSAidGVtcCIpICU+JSAKIyAgIGZpbHRlcihwYXJhbWV0ZXIgJWluJSBjKCJmaWZ0eS1zaXhfZGF5X21lYW4iLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZpZnR5LXNpeF9kYXlfbWVkIiwKIyAgICAgICAgICAgICAgICAgICAgICAgICAgICJmaWZ0eS1zaXhfZGF5X21heCIsIAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZpZnR5LXNpeF9kYXlfbWluIiwgCiMgICAgICAgICAgICAgICAgICAgICAgICAgICAiZmlmdHktc2l4X2RheV92YXIiLAojICAgICAgICAgICAgICAgICAgICAgICAgICAgImZpZnR5LXNpeF9kYXlfcmFuZ2UiKSkgJT4lIAojICAgbXV0YXRlKHBhcmFtZXRlciA9IHBhc3RlKHdvcmQocGFyYW1ldGVyLCBzdGFydCA9IDMsIHNlcCA9IGZpeGVkKCJfIikpLCAiX3RlbXAiLCBzZXAgPSAiIikpICU+JSAKIyAgIGdncGxvdChhZXMoeCA9IGRhdGUsIHkgPSB0ZW1wLCBjb2xvdXIgPSBwYXJhbWV0ZXIpKSArIAojICAgZ2VvbV9saW5lKGxpbmV3aWR0aCA9IDEpICsgCiMgICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IGMoCiMgICAgICJtZWFuX3RlbXAiID0gIm9saXZlZHJhYjMiLAojICAgICAibWVkX3RlbXAiID0gInNlYWdyZWVuMyIsCiMgICAgICJtYXhfdGVtcCIgPSAidG9tYXRvIiwgIAojICAgICAibWluX3RlbXAiID0gImRvZGdlcmJsdWUiLAojICAgICAicmFuZ2VfdGVtcCIgPSAiZ29sZGVucm9kMyIsCiMgICAgICJ2YXJfdGVtcCIgPSAiZGFya2dvbGRlbnJvZDEiCiMgICApKSArIAojICAgc2NhbGVfeF9jb250aW51b3VzKGJyZWFrcyA9IGFzLkRhdGUoYygiMjAyMy0wMS0wMSIsICIyMDIzLTA0LTAxIiwgIjIwMjMtMDctMDEiKSkpICsgCiMgICBnZ3RpdGxlKCJFaWdodCBXZWVrcyIpICsgCiMgICBsYWJzKHkgPSAiVGVtcGVyYXR1cmUgKMKwQykiLAojICAgICAgICB4ID0gIiIpICsgCiMgICB0aGVtZV9idyhiYXNlX3NpemUgPSAyMCkgKyAKIyAgIHRoZW1lKHBhbmVsLmdyaWQgPSBlbGVtZW50X2JsYW5rKCksCiMgICAgICAgICBheGlzLnRleHQueCA9IGVsZW1lbnRfdGV4dChhbmdsZSA9IDI3MCwgaGp1c3QgPSAwLCB2anVzdCA9IDAuNSkpCiMgCiMgZ2dhcnJhbmdlKGRhaWx5X3Bsb3QsIHdlZWtfcGxvdCwgdHdvX3dlZWtfcGxvdCwgZm91cl93ZWVrX3Bsb3QsIGVpZ2h0X3dlZWtfcGxvdCwgCiMgICAgICAgICAgIGNvbW1vbi5sZWdlbmQgPSBULCBucm93ID0gMSwgbGVnZW5kID0gImJvdHRvbSIpCmBgYAoKVGhlIGRpZmZlcmVudCB0aW1lIHBlcmlvZHMgZXhhbWluZWQgYnkgdGhpcyBjbGltYXRlIGRhdGEgaGlnaGxpZ2h0cyB0aGF0IHRoZSByZWxhdGlvbnNoaXAgYmV0d2VlbiBtaW5pbXVtIGFuZCBtYXhpbXVtIHRlbXBlcmF0dXJlcyBjaGFuZ2VzIGJhc2VkIG9uIHRoZSB3aW5kb3cgZXhhbWluZWQuIEZvciBleGFtcGxlLCBtaW5pbXVtIGFuZCBtYXhpbXVtIHRlbXBlcmF0dXJlcyBleHBlcmllbmNlZCBvdmVyIHdlZWtseSBpbnRlcnZhbHMgYXJlIGNsb3NlbHkgbGlua2VkLCB3aGVyZWFzIHRoZXJlIGlzIGEgZGlzdGluY3Qgc2Vhc29uYWwgY3ljbGUgaW4gdGhlIHJlbGF0aW9uc2hpcCBiZXR3ZWVuIG1pbmltdW0gYW5kIG1heGltdW0gdGVtcGVyYXR1cmVzIGV4cGVyaWVuY2VkIG92ZXIgcGVyaW9kcyBvZiBmb3VyIHdlZWtzLiAKCmBgYHtyfQpvbmVfd2Vla19kb3lfZGF0YSA9IHdlZWtfdGVtcHMgJT4lIAogIG11dGF0ZShkb3kgPSB5ZGF5KGRhdGUpKQoKb25lX3dlZWtfdGVtcF9jaXJjbGUgPSBnZ3Bsb3Qob25lX3dlZWtfZG95X2RhdGEsIGFlcyh4ID0gc2V2ZW5fZGF5X21lYW5fbWF4LCB5ID0gc2V2ZW5fZGF5X21lYW5fbWluLCBjb2xvdXIgPSBkb3kpKSArIAogIGdlb21fcG9pbnQoKSArIAogIHNjYWxlX2NvbG91cl9ncmFkaWVudCgKICAgIGhpZ2ggPSAiY29yYWwyIiwKICAgIGxvdyA9ICJkb2RnZXJibHVlNCIpICsgCiAgbGFicyh4ID0gIk1heC4gVGVtcC4gKMKwQykiLAogICAgICAgeSA9ICJNaW4uIFRlbXAuICjCsEMpIikgKyAKICBsYWJzKHggPSAiTWF4LiBUZW1wLiAowrBDKSIsCiAgICAgICB5ID0gIk1pbi4gVGVtcC4gKMKwQykiKSArIAogIGdndGl0bGUoIk9uZSBXZWVrIikgKyAKICB0aGVtZV9tYXR0KCkKCmZvdXJfd2Vla19kb3lfZGF0YSA9IGZvdXJfd2Vla190ZW1wcyAlPiUgCiAgbXV0YXRlKGRveSA9IHlkYXkoZGF0ZSkpCgpmb3VyX3dlZWtfdGVtcF9jaXJjbGUgPSBnZ3Bsb3QoZm91cl93ZWVrX2RveV9kYXRhLCBhZXMoeCA9IGB0d2VudHktZWlnaHRfZGF5X21heGAsIHkgPSBgdHdlbnR5LWVpZ2h0X2RheV9taW5gLCBjb2xvdXIgPSBkb3kpKSArIAogIGdlb21fcG9pbnQoKSArIAogIHNjYWxlX2NvbG91cl9ncmFkaWVudCgKICAgIGhpZ2ggPSAiY29yYWwyIiwKICAgIGxvdyA9ICJkb2RnZXJibHVlNCIpICsgCiAgbGFicyh4ID0gIk1heC4gVGVtcC4gKMKwQykiLAogICAgICAgeSA9ICJNaW4uIFRlbXAuICjCsEMpIikgKyAKICBnZ3RpdGxlKCJGb3VyIFdlZWsiKSArIAogIHRoZW1lX21hdHQoKQoKZ2dhcnJhbmdlKG9uZV93ZWVrX3RlbXBfY2lyY2xlLCBmb3VyX3dlZWtfdGVtcF9jaXJjbGUsCiAgICAgICAgICBjb21tb24ubGVnZW5kID0gVCwgbGVnZW5kID0gImJvdHRvbSIpCmBgYAoKCmBgYHtyIGN0bWF4LXRpbWVzZXJpZXMsIGZpZy53aWR0aD0xMCwgZmlnLmhlaWdodD01fQojIyBEYWlseSB2YWx1ZXMgZm9yIHRoZSBwZXJpb2QgZXhhbWluZWQgYnkgZGF0YXNldApjb2xsZWN0aW9uX2NvbmRpdGlvbnMgPSB0ZW1wX2RhdGEgJT4lCiAgdW5ncm91cCgpICU+JSAKICBncm91cF9ieShkYXRlKSAlPiUgCiAgc3VtbWFyaXNlKG1lYW5fdGVtcCA9IG1lYW4odGVtcCksCiAgICAgICAgICAgIG1lZF90ZW1wID0gbWVkaWFuKHRlbXApLAogICAgICAgICAgICB2YXJfdGVtcCA9IHZhcih0ZW1wKSwgCiAgICAgICAgICAgIG1pbl90ZW1wID0gbWluKHRlbXApLCAKICAgICAgICAgICAgbWF4X3RlbXAgPSBtYXgodGVtcCkpICU+JSAKICBtdXRhdGUoInJhbmdlX3RlbXAiID0gbWF4X3RlbXAgLSBtaW5fdGVtcCwKICAgICAgICAgZGF0ZSA9IGFzLkRhdGUoZGF0ZSkpICU+JSAKICB1bmdyb3VwKCkgJT4lICAKICBmaWx0ZXIoZGF0ZSA+PSAobWluKGFzLkRhdGUoZnVsbF9kYXRhJGNvbGxlY3Rpb25fZGF0ZSkpIC0gNykpCgojIyBNZWFuIGZlbWFsZSB0aGVybWFsIGxpbWl0cyBmb3IgZWFjaCBzcGVjaWVzLCBncm91cGVkIGJ5IGNvbGxlY3Rpb24Kc3BlY2llc19zdW1tYXJpZXMgPSBmdWxsX2RhdGEgJT4lICAKICAjZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUsIGNvbGxlY3Rpb25fZGF0ZSwgY29sbGVjdGlvbl90ZW1wKSAlPiUgIAogIHN1bW1hcmlzZSgibWVhbl9jdG1heCIgPSBtZWFuKGN0bWF4KSwKICAgICAgICAgICAgInNhbXBsZV9zaXplIiA9IG4oKSwKICAgICAgICAgICAgImN0bWF4X3N0X2VyciIgPSAoc2QoY3RtYXgpIC8gc3FydChzYW1wbGVfc2l6ZSkpLAogICAgICAgICAgICAiY3RtYXhfdmFyIiA9IHZhcihjdG1heCksIAogICAgICAgICAgICAibWVhbl9zaXplIiA9IG1lYW4oc2l6ZSksCiAgICAgICAgICAgICJzaXplX3N0X2VyciIgPSAoc2Qoc2l6ZSkgLyBzcXJ0KHNhbXBsZV9zaXplKSksCiAgICAgICAgICAgICJzaXplX3ZhciIgPSB2YXIoc2l6ZSkpICU+JSAgCiAgdW5ncm91cCgpICU+JSAKICBjb21wbGV0ZShzcF9uYW1lLCBjb2xsZWN0aW9uX2RhdGUpICU+JSAKICBhcnJhbmdlKGRlc2Moc2FtcGxlX3NpemUpKQoKYWR1bHRfc3VtbWFyaWVzID0gZnVsbF9kYXRhICU+JSAgCiAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAogIGdyb3VwX2J5KHNwX25hbWUsIGNvbGxlY3Rpb25fZGF0ZSwgY29sbGVjdGlvbl90ZW1wKSAlPiUgIAogIHN1bW1hcmlzZSgibWVhbl9jdG1heCIgPSBtZWFuKGN0bWF4KSwKICAgICAgICAgICAgInNhbXBsZV9zaXplIiA9IG4oKSwKICAgICAgICAgICAgImN0bWF4X3N0X2VyciIgPSAoc2QoY3RtYXgpIC8gc3FydChzYW1wbGVfc2l6ZSkpLAogICAgICAgICAgICAiY3RtYXhfdmFyIiA9IHZhcihjdG1heCksIAogICAgICAgICAgICAibWVhbl9zaXplIiA9IG1lYW4oc2l6ZSksCiAgICAgICAgICAgICJzaXplX3N0X2VyciIgPSAoc2Qoc2l6ZSkgLyBzcXJ0KHNhbXBsZV9zaXplKSksCiAgICAgICAgICAgICJzaXplX3ZhciIgPSB2YXIoc2l6ZSkpICU+JSAgCiAgdW5ncm91cCgpICU+JSAKICBjb21wbGV0ZShzcF9uYW1lLCBjb2xsZWN0aW9uX2RhdGUpICU+JSAKICBhcnJhbmdlKGRlc2Moc2FtcGxlX3NpemUpKQoKZ2dwbG90KCkgKyAKICBnZW9tX3ZsaW5lKGRhdGEgPSB1bmlxdWUoc2VsZWN0KGZ1bGxfZGF0YSwgY29sbGVjdGlvbl9kYXRlKSksIAogICAgICAgICAgICAgYWVzKHhpbnRlcmNlcHQgPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSkpLAogICAgICAgICAgICAgY29sb3VyID0gImdyZXk5MCIsCiAgICAgICAgICAgICBsaW5ld2lkdGggPSAxKSArIAogIGdlb21fbGluZShkYXRhID0gY29sbGVjdGlvbl9jb25kaXRpb25zLCAKICAgICAgICAgICAgYWVzKHggPSBhcy5EYXRlKGRhdGUpLCB5ID0gbWVhbl90ZW1wKSwKICAgICAgICAgICAgY29sb3VyID0gImJsYWNrIiwgCiAgICAgICAgICAgIGxpbmV3aWR0aCA9IDIpICsgCiAgIyBnZW9tX2Vycm9yYmFyKGRhdGEgPSBzcGVjaWVzX3N1bW1hcmllcywKICAjICAgICAgICAgICAgICAgYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksCiAgIyAgICAgICAgICAgICAgICAgICB5bWluID0gbWVhbl9jdG1heCAtIGN0bWF4X3N0X2VyciwgeW1heCA9IG1lYW5fY3RtYXggKyBjdG1heF9zdF9lcnIsCiAgIyAgICAgICAgICAgICAgICAgICBjb2xvdXIgPSBzcF9uYW1lKSwKICAjICAgICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9kb2RnZSh3aWR0aCA9IDEpLAogICMgICAgICAgICAgICAgICB3aWR0aCA9IDUsIGxpbmV3aWR0aCA9IDEpICsKICBnZW9tX3BvaW50KGRhdGEgPSBzcGVjaWVzX3N1bW1hcmllcywgCiAgICAgICAgICAgICBhZXMoeCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSwgeSA9IG1lYW5fY3RtYXgsIGNvbG91ciA9IHNwX25hbWUsIHNpemUgPSBzYW1wbGVfc2l6ZSkpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgbGFicyh4ID0gIkRhdGUiLCAKICAgICAgIHkgPSAiVGVtcGVyYXR1cmUgKMKwQykiLCAKICAgICAgIGNvbG91ciA9ICJTcGVjaWVzIiwKICAgICAgIHNpemUgPSAiU2FtcGxlIFNpemUiKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCgpzaXplX3RpbWVzZXJpZXMgPSBnZ3Bsb3QoKSArIAogIGdlb21fdmxpbmUoZGF0YSA9IHVuaXF1ZShzZWxlY3QoZnVsbF9kYXRhLCBjb2xsZWN0aW9uX2RhdGUpKSwgCiAgICAgICAgICAgICBhZXMoeGludGVyY2VwdCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSksCiAgICAgICAgICAgICBjb2xvdXIgPSAiZ3JleTkwIiwKICAgICAgICAgICAgIGxpbmV3aWR0aCA9IDEpICsgCiAgZ2VvbV9saW5lKGRhdGEgPSBjb2xsZWN0aW9uX2NvbmRpdGlvbnMsIAogICAgICAgICAgICBhZXMoeCA9IGFzLkRhdGUoZGF0ZSksIHkgPSBtZWFuX3RlbXApLAogICAgICAgICAgICBjb2xvdXIgPSAiYmxhY2siLCAKICAgICAgICAgICAgbGluZXdpZHRoID0gMikgKyAKICAjIGdlb21fZXJyb3JiYXIoZGF0YSA9IHNwZWNpZXNfc3VtbWFyaWVzLAogICMgICAgICAgICAgICAgICBhZXMoeCA9IGFzLkRhdGUoY29sbGVjdGlvbl9kYXRlKSwgCiAgIyAgICAgICAgICAgICAgICAgICB5bWluID0gbWVhbl9jdG1heCAtIGN0bWF4X3N0X2VyciwgeW1heCA9IG1lYW5fY3RtYXggKyBjdG1heF9zdF9lcnIsCiAgIyAgICAgICAgICAgICAgICAgICBjb2xvdXIgPSBzcF9uYW1lKSwKICAjICAgICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9kb2RnZSh3aWR0aCA9IDEpLAogICMgICAgICAgICAgICAgICB3aWR0aCA9IDUsIGxpbmV3aWR0aCA9IDEpICsgCiAgZ2VvbV9wb2ludChkYXRhID0gc3BlY2llc19zdW1tYXJpZXMsIAogICAgICAgICAgICAgYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksIHkgPSBtZWFuX3NpemUgKiA0MCwgY29sb3VyID0gc3BfbmFtZSksCiAgICAgICAgICAgICBwb3NpdGlvbiA9IHBvc2l0aW9uX2RvZGdlKHdpZHRoID0gMSksCiAgICAgICAgICAgICBzaXplID0gNCkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBzY2FsZV95X2NvbnRpbnVvdXMoCiAgICBuYW1lID0gIlRlbXBlcmF0dXJlIiwgIyBGZWF0dXJlcyBvZiB0aGUgZmlyc3QgYXhpcwogICAgc2VjLmF4aXMgPSBzZWNfYXhpcyh+Li80MCwgbmFtZT0iUHJvc29tZSBMZW5ndGggKG1tKSIpLCAjIEFkZCBhIHNlY29uZCBheGlzIGFuZCBzcGVjaWZ5IGl0cyBmZWF0dXJlcwogICAgYnJlYWtzID0gYygwLDUsMTAsMTUsMjAsMjUsMzApCiAgKSArIAogIGxhYnMoeCA9ICJEYXRlIiwgCiAgICAgICB5ID0gIlRlbXBlcmF0dXJlICjCsEMpIiwgCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKCiNnZ2FycmFuZ2UoY3RtYXhfdGltZXNlcmllcywgc2l6ZV90aW1lc2VyaWVzLCBjb21tb24ubGVnZW5kID0gVCwgbGVnZW5kID0gImJvdHRvbSIpCmBgYAoKCmBgYHtyIHB1bGxpbmctcHJlZGljdG9yc30KIyAjIyBDb21iaW5lIGRhdGEsIHRoZW4gcHVsbCBvdXQgdmFsdWVzIGZvciBlYWNoIGNvbGxlY3Rpb24gZGF0ZQojIGRhdGVfbGlzdCA9IGFzLkRhdGUodW5pcXVlKGZ1bGxfZGF0YSRjb2xsZWN0aW9uX2RhdGUpKQojIAojIHRlbXBfcHJlZGljdG9ycyA9IGRhaWx5X3RlbXBfZGF0YSAlPiUgCiMgICBmdWxsX2pvaW4oZGF5X3ByaW9yX3RlbXBfZGF0YSwgYnkgPSBjKCJkYXRlIikpICU+JSAKIyAgIGZ1bGxfam9pbih0aHJlZV9kYXlfdGVtcHMsIGJ5ID0gYygiZGF0ZSIpKSAlPiUgCiMgICBmdWxsX2pvaW4od2Vla190ZW1wcywgYnkgPSBjKCJkYXRlIikpICU+JSAKIyAgIGZ1bGxfam9pbih0d29fd2Vla190ZW1wcywgYnkgPSBjKCJkYXRlIikpICU+JSAKIyAgIGZ1bGxfam9pbihmb3VyX3dlZWtfdGVtcHMsIGJ5ID0gYygiZGF0ZSIpKSAlPiUgCiMgICBmdWxsX2pvaW4oZWlnaHRfd2Vla190ZW1wcywgYnkgPSBjKCJkYXRlIikpICU+JSAKIyAgIGZpbHRlcihkYXRlICVpbiUgZGF0ZV9saXN0KQpgYGAKCkEgc2V0IG9mIHByZWRpY3RvcnMgdmFyaWFibGVzIHdlcmUgYXNzZW1ibGVkIGZyb20gdGhlIGNvbnRpbnVvdXMgdGVtcGVyYXR1cmUgZGF0YSBzZXQgYmFzZWQgb24gY29uZGl0aW9ucyBkdXJpbmcgdGhlIGRheSBvZiBjb2xsZWN0aW9uLCB0aGUgd2VlayBiZWZvcmUgY29sbGVjdGlvbnMsIGFuZCB0aGUgcHJlY2VkaW5nIHR3bywgZm91ciwgYW5kIGVpZ2h0IHdlZWsgcGVyaW9kcy4gVGhpcyBpcyBhIHByZWxpbWluYXJ5IGFuYWx5c2lzIGZvciBub3cuIFNob3duIGhlcmUgYXJlIHRoZSB0b3AgdGhyZWUgZmFjdG9ycy4gU3BlY2llcyB3aXRoIG5vIHNpZ25pZmljYW50IHByZWRpY3RvciBvciBsaW1pdGVkIGNvbGxlY3Rpb24gZGF0ZSBkaXN0cmlidXRpb25zIHdlcmUgZXhjbHVkZWQuIAoKYGBge3IgcHJlZGljdG9yLWNvcnJlbGF0aW9uc30KIyAKIyBjb3JyX3ZhbHMgPSBmdWxsX2RhdGEgJT4lCiMgICBmaWx0ZXIoc3BfbmFtZSAlaW4lIG51bV9jb2xscyRzcF9uYW1lKSAlPiUgCiMgICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgCiMgICBtdXRhdGUoY29sbGVjdGlvbl9kYXRlID0gYXMuRGF0ZShjb2xsZWN0aW9uX2RhdGUpKSAlPiUgCiMgICBmdWxsX2pvaW4odGVtcF9wcmVkaWN0b3JzLCBqb2luX2J5KGNvbGxlY3Rpb25fZGF0ZSA9PSBkYXRlKSkgJT4lIAojICAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKGNvbGxlY3Rpb25fdGVtcCwgbWVhbl90ZW1wOnRhaWwobmFtZXMoLiksIDEpKSwKIyAgICAgICAgICAgICAgICB2YWx1ZXNfdG8gPSAidmFsdWUiLCAKIyAgICAgICAgICAgICAgICBuYW1lc190byA9ICJwcmVkaWN0b3IiKSAlPiUgIAojICAgZ3JvdXBfYnkoc3BfbmFtZSwgcHJlZGljdG9yKSAlPiUgCiMgICBzdW1tYXJpc2UoY29ycmVsYXRpb24gPSBjb3IudGVzdChjdG1heCwgdmFsdWUpJGVzdGltYXRlLAojICAgICAgICAgICAgIHAudmFsdWUgPSBjb3IudGVzdChjdG1heCwgdmFsdWUpJHAudmFsdWUsCiMgICAgICAgICAgICAgY2lfbG93ID0gY29yLnRlc3QoY3RtYXgsIHZhbHVlKSRjb25mLmludFsxXSwKIyAgICAgICAgICAgICBjaV9oaWdoID0gY29yLnRlc3QoY3RtYXgsIHZhbHVlKSRjb25mLmludFsyXSkgJT4lIAojICAgbXV0YXRlKHNpZyA9IGlmZWxzZShwLnZhbHVlIDwwLjA1LCAiU2lnLiIsICJOb24gU2lnLiIpKQoKY29ycl92YWxzICU+JSAgCiAgZmlsdGVyKHNpZyA9PSAiU2lnLiIpICU+JSAKICBkcm9wX25hKGNvcnJlbGF0aW9uKSAlPiUgCiAgZ3JvdXBfYnkoc3BfbmFtZSkgJT4lCiAgYXJyYW5nZShkZXNjKGNvcnJlbGF0aW9uKSkgJT4lIAogIHNsaWNlX2hlYWQobiA9IDMpICU+JSAKICBzZWxlY3QoIlNwZWNpZXMiID0gc3BfbmFtZSwgIlByZWRpY3RvciIgPSBwYXJhbWV0ZXIsICJEdXJhdGlvbiIgPSBkdXJhdGlvbiwgIkNvcnJlbGF0aW9uIiA9IGNvcnJlbGF0aW9uLCAiUC1WYWx1ZSIgPSBwLnZhbHVlKSAlPiUgCiAga25pdHI6OmthYmxlKGFsaWduID0gImMiKQpgYGAKCiMjIFRyYWl0IFZhcmlhdGlvbiAKYGBge3IgY3RtYXgtYW5kLXNpemUtc3VtLXBsb3QsIGZpZy53aWR0aD0yMCwgZmlnLmhlaWdodD01fQpjdG1heF9wbG90ID0gZnVsbF9kYXRhICU+JSAKICBtdXRhdGUoICNzcF9uYW1lID0gc3RyX3JlcGxhY2Uoc3BfbmFtZSwgcGF0dGVybiA9ICIgIiwKICAgICMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXBsYWNlbWVudCA9ICJcbiIpLAogICAgc3BfbmFtZSA9IGZjdF9yZW9yZGVyKHNwX25hbWUsIGN0bWF4LCBtZWFuKSkgJT4lIAogIGdncGxvdChhZXMoeSA9IHNwX25hbWUsIHggPSBjdG1heCkpICsgCiAgZ2VvbV9wb2ludChhZXMoY29sb3VyPSBzcF9uYW1lX3N1YiksCiAgICAgICAgICAgICBwb3NpdGlvbiA9IHBvc2l0aW9uX2RvZGdlKHdpZHRoID0gMC4zKSwKICAgICAgICAgICAgIHNpemUgPSA0KSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHhsYWIoTlVMTCkgKyAKICBsYWJzKHkgPSAiIiwKICAgICAgIHggPSAiQ1RtYXggKMKwQykiLAogICAgICAgY29sb3VyID0gIkdyb3VwIikgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCgpzaXplX3Bsb3QgPSBmdWxsX2RhdGEgJT4lIAogIG11dGF0ZShzcF9uYW1lID0gZmN0X3Jlb3JkZXIoc3BfbmFtZSwgY3RtYXgsIG1lYW4pKSAlPiUgCiAgZ2dwbG90KGFlcyh5ID0gc3BfbmFtZSwgeCA9IHNpemUpKSArIAogIGdlb21fcG9pbnQoYWVzKGNvbG91cj0gc3BfbmFtZV9zdWIpLAogICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9kb2RnZSh3aWR0aCA9IDAuMyksCiAgICAgICAgICAgICBzaXplID0gNCkgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiUHJvc29tZSBMZW5ndGggKG1tKSIsCiAgICAgICB5ID0gIiIsIAogICAgICAgY29sb3VyID0gIkdyb3VwIikgKyAKICBndWlkZXMoY29sb3IgPSBndWlkZV9sZWdlbmQobmNvbCA9IDEpKSArCiAgdGhlbWVfbWF0dChiYXNlX3NpemUgPSApICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IiwKICAgICAgICBheGlzLnRleHQueSA9IGVsZW1lbnRfYmxhbmsoKSwKICAgICAgICBwbG90Lm1hcmdpbiA9IG1hcmdpbigwLCAwLCAwLCAwLCJjbSIpKQoKdHJhaXRfcGxvdCA9IGN0bWF4X3Bsb3QgKyBzaXplX3Bsb3QKdHJhaXRfcGxvdApgYGAKCmBgYHtyIGN0bWF4LWFuZC1zaXplLWhpc3RvZ3JhbXMsIGZpZy53aWR0aD03LCBmaWcuaGVpZ2h0PTIxLCBpbmNsdWRlID0gRn0KZ2dwbG90KGZ1bGxfZGF0YSwgYWVzKHggPSBzaXplLCBmaWxsID0gc3BfbmFtZV9zdWIpKSArIAogIGZhY2V0X3dyYXAoLn5zcF9uYW1lX3N1YiwgbmNvbCA9IDEpICsgCiAgZ2VvbV9oaXN0b2dyYW0oYmlud2lkdGggPSAwLjA1KSArIAogIHNjYWxlX2ZpbGxfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiUHJvc29tZSBsZW5ndGggKG1tKSIpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKCmdncGxvdChmdWxsX2RhdGEsIGFlcyh4ID0gY3RtYXgsIGZpbGwgPSBzcF9uYW1lX3N1YikpICsgCiAgZmFjZXRfd3JhcCgufnNwX25hbWVfc3ViLCBuY29sID0gMSkgKyAKICBnZW9tX2hpc3RvZ3JhbShiaW53aWR0aCA9IDEpICsgCiAgc2NhbGVfZmlsbF9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIGxhYnMoeCA9ICJDVG1heCAowrBDKSIpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgpgYGB7ciBmZWN1bmRpdHktaGlzdG9ncmFtLCBmaWcud2lkdGg9NywgZmlnLmhlaWdodD0xMH0KZnVsbF9kYXRhICU+JSAgCiAgZHJvcF9uYShmZWN1bmRpdHkpICU+JSAgCiAgZ2dwbG90KGFlcyh4ID0gZmVjdW5kaXR5LCBmaWxsID0gc3BfbmFtZV9zdWIpKSArIAogIGZhY2V0X3dyYXAoLn5zcF9uYW1lX3N1YiwgbmNvbCA9IDEpICsgCiAgZ2VvbV9oaXN0b2dyYW0oYmlud2lkdGggPSAyKSArIAogIHNjYWxlX2ZpbGxfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICBsYWJzKHggPSAiRmVjdW5kaXR5ICgjIEVnZ3MpIikgKwogIHRoZW1lX21hdHRfZmFjZXRzKCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCmBgYAoKIyMjIFZhcmlhdGlvbiB3aXRoIHRlbXBlcmF0dXJlIAoKYGBge3IgdHJhaXQtY29sbC10ZW1wLXBsb3RzLCBmaWcud2lkdGg9MTUsIGZpZy5oZWlnaHQ9MTB9CmN0bWF4X3RlbXAgPSBnZ3Bsb3QoZnVsbF9kYXRhLCBhZXMoeCA9IGNvbGxlY3Rpb25fdGVtcCwgeSA9IGN0bWF4LCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBsaW5ld2lkdGggPSAzKSArCiAgZ2VvbV9wb2ludChzaXplID0gMykgKyAKICBsYWJzKHggPSAiQ29sbGVjdGlvbiBUZW1wZXJhdHVyZSAowrBDKSIsIAogICAgICAgeSA9ICJDVG1heCAowrBDKSIsCiAgICAgICBjb2xvdXIgPSAiU3BlY2llcyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKCnNpemVfdGVtcCA9IGdncGxvdChmaWx0ZXIoZnVsbF9kYXRhLCBzZXggIT0gImp1dmVuaWxlIiksIGFlcyh4ID0gY29sbGVjdGlvbl90ZW1wLCB5ID0gc2l6ZSwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgbGluZXdpZHRoID0gMykgKwogIGdlb21fcG9pbnQoc2l6ZSA9IDMpICsgCiAgbGFicyh4ID0gIkNvbGxlY3Rpb24gVGVtcGVyYXR1cmUgKMKwQykiLCAKICAgICAgIHkgPSAiTGVuZ3RoIChtbSkiLAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSAgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiKQoKd3RfdGVtcCA9IGdncGxvdChmdWxsX2RhdGEsIGFlcyh4ID0gY29sbGVjdGlvbl90ZW1wLCB5ID0gd2FybWluZ190b2wsIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIGxpbmV3aWR0aCA9IDMpICsKICBnZW9tX3BvaW50KHNpemUgPSAzKSArIAogIGxhYnMoeCA9ICJDb2xsZWN0aW9uIFRlbXBlcmF0dXJlICjCsEMpIiwgCiAgICAgICB5ID0gIldhcm1pbmcgVG9sZXJhbmNlICjCsEMpIiwKICAgICAgIGNvbG91ciA9ICJTcGVjaWVzIikgICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKCmVnZ3NfdGVtcCA9IGdncGxvdChmdWxsX2RhdGEsIGFlcyh4ID0gY29sbGVjdGlvbl90ZW1wLCB5ID0gZmVjdW5kaXR5LCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBsaW5ld2lkdGggPSAzKSArCiAgZ2VvbV9wb2ludChzaXplID0gMykgKyAKICBsYWJzKHggPSAiQ29sbGVjdGlvbiBUZW1wZXJhdHVyZSAowrBDKSIsIAogICAgICAgeSA9ICJGZWN1bmRpdHkgKCMgRWdncykiLAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSAgKyAKICBzY2FsZV9jb2xvdXJfbWFudWFsKHZhbHVlcyA9IHNwZWNpZXNfY29scykgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiKQoKZ2dhcnJhbmdlKGN0bWF4X3RlbXAsIHNpemVfdGVtcCwgd3RfdGVtcCwgZWdnc190ZW1wLCAKICAgICAgICAgIGNvbW1vbi5sZWdlbmQgPSBULCBsZWdlbmQgPSAicmlnaHQiKQpgYGAKCmBgYHtyIGN0bWF4LWNvbGwtdGVtcC1tb2RlbH0KIyBhZHVsdF9kYXRhID0gZnVsbF9kYXRhICU+JSAKIyAgIGZpbHRlcihzZXggPT0gImZlbWFsZSIpCm1vZGVsX2RhdGEgPSBmdWxsX2RhdGEgJT4lICAKICBkcm9wX25hKHNpemUsIGN0bWF4KQoKY3RtYXhfdGVtcC5tb2RlbCA9IGxtKGRhdGEgPSBtb2RlbF9kYXRhLCBjdG1heCB+IGNvbGxlY3Rpb25fdGVtcCAqIHNwX25hbWUgKyBzZXgpCnNpemVfdGVtcC5tb2RlbCA9IGxtKGRhdGEgPSBtb2RlbF9kYXRhLCBzaXplIH4gY29sbGVjdGlvbl90ZW1wICogc3BfbmFtZSArIHNleCkKCmtuaXRyOjprYWJsZShjYXI6OkFub3ZhKGN0bWF4X3RlbXAubW9kZWwpKQoKY3RtYXhfcmVzaWRzID0gY2JpbmQobW9kZWxfZGF0YSwgInJlc2lkcyIgPSBjdG1heF90ZW1wLm1vZGVsJHJlc2lkdWFscywgInNpemVfcmVzaWRzIiA9IHNpemVfdGVtcC5tb2RlbCRyZXNpZHVhbHMpCmBgYAoKYGBge3IgY3RtYXgtdGltZS1pbi1sYWIsIGZpZy53aWR0aD0xNSwgZmlnLmhlaWdodD0xMH0KZ2dwbG90KGN0bWF4X3Jlc2lkcywgYWVzKHggPSBkYXlzX2luX2xhYiwgeSA9IHJlc2lkcywgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZmFjZXRfd3JhcChzcF9uYW1lfi4pICsgCiAgZ2VvbV9wb2ludChzaXplID0gNCwgYWxwaGEgPSAwLjUpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGLCBsaW5ld2lkdGggPSAyKSArIAogIHNjYWxlX3hfY29udGludW91cyhicmVha3MgPSBjKDA6NSkpICsgCiAgbGFicyh4ID0gIkRheXMgaW4gbGFiIiwgCiAgICAgICB5ID0gIkNUbWF4IFJlc2lkdWFscyIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgpHaXZlbiB0aGUgbG9uZyBnZW5lcmF0aW9uIHRpbWVzIG9mIHRoZXNlIGNvcGVwb2RzLCBkZWNyZWFzZXMgaW4gdHJhaXQgdmFyaWFuY2UgbWF5IGluZGljYXRlIHNlbGVjdGlvbiBvdmVyIHRoZSBzZWFzb25hbCBjeWNsZS4gU2hvd24gYmVsb3cgYXJlIHRoZSB2YXJpYW5jZSBpbiBvYnNlcnZlZCBDVG1heCBhbmQgc2l6ZSwgcGxvdHRlZCBhZ2FpbnN0IGNvbGxlY3Rpb24gZGF0ZS4gVmFyaWFuY2UgZGVjcmVhc2VzIGluICpTa2lzdG9kaWFwdG9tdXMqLCBidXQgdGhpcyBwYXR0ZXJuIGlzIGRyaXZlbiBieSBhIHNpbmdsZSBjb2xsZWN0aW9uIHdpdGggaGlnaCB2YXJpYW5jZSBlYXJseSBpbiB0aGUgeWVhci4gU2l6ZSB2YXJpYW5jZSBpbmNyZWFzZXMgc2xpZ2h0bHkgaW4gKlNraXN0b2RpYXB0b211cyouIFZhcmlhbmNlIGluIGJvdGggQ1RtYXggYW5kIHNpemUgaXMgZmFpcmx5IGNvbnN0YW50IGluICpMZXB0b2RpYXB0b211cyBtaW51dHVzKiwgdGhlIG9ubHkgb3RoZXIgc3BlY2llcyBjb2xsZWN0ZWQgYWNyb3NzIHRoZSBlbnRpcmUgc2V0IG9mIHNhbXBsZXMgdGh1cyBmYXIuIAoKYGBge3IgdHJhaXQtdmFyaWFuY2UtY29sbC10ZW1wfQpnZ3Bsb3QoZHJvcF9uYShhZHVsdF9zdW1tYXJpZXMsIGN0bWF4X3ZhciksIGFlcyh4ID0gYXMuRGF0ZShjb2xsZWN0aW9uX2RhdGUpLCB5ID0gY3RtYXhfdmFyLCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBmYWNldF93cmFwKHNwX25hbWV+Liwgc2NhbGVzID0gImZyZWVfeSIpICsgCiAgZ2VvbV9wb2ludChzaXplID0gMikgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYpICsgCiAgbGFicyh4ID0gIkNvbGxlY3Rpb24gVGVtcC4gKMKwQykiLCAKICAgICAgIHkgPSAiQ1RtYXggVmFyaWFuY2UiKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX21hdHRfZmFjZXRzKCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAibm9uZSIpCgpnZ3Bsb3QoZHJvcF9uYShhZHVsdF9zdW1tYXJpZXMsIHNpemVfdmFyKSwgYWVzKHggPSBhcy5EYXRlKGNvbGxlY3Rpb25fZGF0ZSksIHkgPSBzaXplX3ZhciwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZmFjZXRfd3JhcChzcF9uYW1lfi4pICsgCiAgZ2VvbV9wb2ludChzaXplID0gMikgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYpICsgCiAgbGFicyh4ID0gIkNvbGxlY3Rpb24gVGVtcC4gKMKwQykiLCAKICAgICAgIHkgPSAiU2l6ZSBWYXJpYW5jZSIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dF9mYWNldHMoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIikKYGBgCgoKIyMgU2V4IGFuZCBzdGFnZSB2YXJpYXRpb24gaW4gdGhlcm1hbCBsaW1pdHMgClByZXZpb3VzIHNlY3Rpb25zIGhhdmUgZ2VuZXJhbGx5IGx1bXBlZCBqdXZlbmlsZSwgZmVtYWxlLCBhbmQgbWFsZSBpbmRpdmlkdWFscyB0b2dldGhlci4gVGhlcmUgbWF5IGJlIGltcG9ydGFudCBzdGFnZS0gb3Igc2V4LXNwZWNpZmljIGRpZmZlcmVuY2VzIGluIENUbWF4IHRob3VnaC4gRm9yIHNldmVyYWwgc3BlY2llcywgd2UgaGF2ZSBtZWFzdXJlbWVudHMgZm9yIGluZGl2aWR1YWxzIGluIGRpZmZlcmVudCBzdGFnZXMgb3Igb2YgZGlmZmVyZW50IHNleGVzLiAKCmBgYHtyIHNleC1zdGFnZS10YWJsZX0Kc2V4X3NhbXBsZV9zaXplcyA9IGN0bWF4X3Jlc2lkcyAlPiUgIAogIGdyb3VwX2J5KHNwX25hbWUsIHNleCkgJT4lICAKICBzdW1tYXJpc2UobnVtID0gbigpKSAlPiUgIAogIHBpdm90X3dpZGVyKGlkX2NvbHMgPSBzcF9uYW1lLAogICAgICAgICAgICAgIG5hbWVzX2Zyb20gPSBzZXgsIAogICAgICAgICAgICAgIHZhbHVlc19mcm9tID0gbnVtLAogICAgICAgICAgICAgIHZhbHVlc19maWxsID0gMCkgJT4lIAogIHNlbGVjdCgiU3BlY2llcyIgPSBzcF9uYW1lLCAiSnV2ZW5pbGUiID0ganV2ZW5pbGUsICJGZW1hbGUiID0gZmVtYWxlLCAiTWFsZSIgPSBtYWxlKQoKa25pdHI6OmthYmxlKHNleF9zYW1wbGVfc2l6ZXMsIGFsaWduID0gImMiKQpgYGAKClRoZSBmZW1hbGUtbWFsZSBhbmQgZmVtYWxlLWp1dmVuaWxlIGNvbXBhcmlzb25zIHNob3cgdGhhdCB0aGVyZSBhcmUgZ2VuZXJhbGx5IG5vIGRpZmZlcmVuY2VzIGluIHRoZXJtYWwgbGltaXRzIGJldHdlZW4gdGhlc2UgZ3JvdXBzLiAKCmBgYHtyIGN0bWF4LXNleCwgZmlnLndpZHRoPTcsIGZpZy5oZWlnaHQ9N30KY3RtYXhfcmVzaWRzICU+JSAKICBmaWx0ZXIoc3BfbmFtZSAlaW4lIGZpbHRlcihzZXhfc2FtcGxlX3NpemVzLCBNYWxlID4gMCwgRmVtYWxlID4gMCkkU3BlY2llcyAmIAogICAgICAgICAgIHNleCAhPSAianV2ZW5pbGUiKSAlPiUgCiAgZ2dwbG90KGFlcyh4ID0gc2V4LCB5ID0gcmVzaWRzLCBjb2xvdXIgPSBzcF9uYW1lLCBncm91cCA9IHNwX25hbWUpKSArIAogIGZhY2V0X3dyYXAoc3BfbmFtZX4uLCBuY29sID0gMikgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYsIGxpbmV3aWR0aCA9IDEpICsgCiAgZ2VvbV9wb2ludChzaXplID0gMywKICAgICAgICAgICAgIGFscGhhID0gMC41LAogICAgICAgICAgICAgcG9zaXRpb24gPSBwb3NpdGlvbl9qaXR0ZXIoaGVpZ2h0ID0gMCwgd2lkdGggPSAwLjA1KSkgKyAgCiAgbGFicyh4ID0gIlNleCIsIAogICAgICAgeSA9ICJDVG1heCBSZXNpZHVhbHMiKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX2J3KGJhc2Vfc2l6ZSA9IDE4KSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIiwgCiAgICAgICAgcGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKSkKYGBgCgpgYGB7ciBjdG1heC1zdGFnZSwgZmlnLndpZHRoPTMsIGZpZy5oZWlnaHQ9Nn0KY3RtYXhfcmVzaWRzICU+JSAKICBmaWx0ZXIoc3BfbmFtZSAlaW4lIGZpbHRlcihzZXhfc2FtcGxlX3NpemVzLCBKdXZlbmlsZSA+IDAgJiBGZW1hbGUgPiAwKSRTcGVjaWVzICYgCiAgICAgICAgICAgc2V4ICE9ICJtYWxlIikgJT4lIAogIGdncGxvdChhZXMoeCA9IHNleCwgeSA9IHJlc2lkcywgY29sb3VyID0gc3BfbmFtZSwgZ3JvdXAgPSBzcF9uYW1lKSkgKyAKICBmYWNldF93cmFwKHNwX25hbWV+LiwgbmNvbCA9IDEpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGLCBsaW5ld2lkdGggPSAxKSArIAogIGdlb21fcG9pbnQoc2l6ZSA9IDMsCiAgICAgICAgICAgICBhbHBoYSA9IDAuNSwKICAgICAgICAgICAgIHBvc2l0aW9uID0gcG9zaXRpb25faml0dGVyKGhlaWdodCA9IDAsIHdpZHRoID0gMC4wNSkpICsgIAogIGxhYnMoeCA9ICJTZXgiLCAKICAgICAgIHkgPSAiQ1RtYXggKMKwQykiKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX2J3KGJhc2Vfc2l6ZSA9IDE4KSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJub25lIiwgCiAgICAgICAgcGFuZWwuZ3JpZCA9IGVsZW1lbnRfYmxhbmsoKSkKYGBgCgojIyBUcmFpdCBDb3JyZWxhdGlvbnMgCgpgYGB7ciBjdG1heC1zaXplLCBmaWcud2lkdGg9MTAsIGZpZy5oZWlnaHQ9N30KCmZ1bGxfZGF0YSAlPiUgCiAgI2ZpbHRlcihzZXggPT0gImZlbWFsZSIpICU+JSAgCiAgZ2dwbG90KCBhZXMoeCA9IHNpemUsIHkgPSBjdG1heCwgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZ2VvbV9zbW9vdGgoZGF0YSA9IGZ1bGxfZGF0YSwgCiAgICAgICAgICAgICAgYWVzKHggPSBzaXplLCB5ID0gY3RtYXgpLAogICAgICAgICAgICAgIG1ldGhvZCA9ICJsbSIsIAogICAgICAgICAgICAgIGNvbG91ciA9ImJsYWNrIiwgCiAgICAgICAgICAgICAgbGluZXdpZHRoID0gMi41KSArIAogIGdlb21fcG9pbnQoc2l6ZSA9IDIsIGFscGhhID0gMC40KSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRiwgbGluZXdpZHRoID0gMikgKyAKICBsYWJzKHggPSAiTGVuZ3RoIChtbSkiLCAKICAgICAgIHkgPSAiQ1RtYXggKMKwQykiLAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCmBgYAoKYGBge3IgY3RtYXhyZXNpZHMtc2l6ZSwgZmlnLndpZHRoPTEwLCBmaWcuaGVpZ2h0PTcsIGluY2x1ZGUgPSBGfQpjdG1heF9yZXNpZHMgJT4lIAogIGZpbHRlcihzZXggPT0gImZlbWFsZSIpICU+JSAKICBnZ3Bsb3QoYWVzKHggPSBzaXplX3Jlc2lkcywgeSA9IHJlc2lkcywgY29sb3VyID0gc3BfbmFtZSkpICsgCiAgZ2VvbV9zbW9vdGgoZGF0YSA9IGN0bWF4X3Jlc2lkcywgCiAgICAgICAgICAgICAgYWVzKHggPSBzaXplX3Jlc2lkcywgeSA9IHJlc2lkcyksCiAgICAgICAgICAgICAgbWV0aG9kID0gImxtIiwgCiAgICAgICAgICAgICAgY29sb3VyID0iYmxhY2siLCAKICAgICAgICAgICAgICBsaW5ld2lkdGggPSAyLjUpICsgCiAgZ2VvbV9wb2ludChzaXplID0gMiwgYWxwaGEgPSAwLjQpICsgCiAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIiwgc2UgPSBGLCBsaW5ld2lkdGggPSAyKSArIAogIGxhYnMoeCA9ICJMZW5ndGggKG1tKSIsIAogICAgICAgeSA9ICJDVG1heCAowrBDKSIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKYGBgCgoKYGBge3IgZmVjdW5kaXR5LXNpemUsIGZpZy53aWR0aD0xMCwgZmlnLmhlaWdodD03fQpnZ3Bsb3QoY3RtYXhfcmVzaWRzLCBhZXMoeCA9IHNpemUsIHkgPSBmZWN1bmRpdHksIGNvbG91ciA9IHNwX25hbWUpKSArIAogIGdlb21fc21vb3RoKG1ldGhvZCA9ICJsbSIsIHNlID0gRiwgbGluZXdpZHRoID0gMikgKyAKICBnZW9tX3BvaW50KHNpemUgPSAyLCBhbHBoYSA9IDAuNSkgKyAKICBsYWJzKHggPSAiUHJvc29tZSBsZW5ndGggKG1tKSIsIAogICAgICAgeSA9ICJGZWN1bmRpdHkgKCMgRWdncykiLAogICAgICAgY29sb3VyID0gIlNwZWNpZXMiKSArIAogIHNjYWxlX2NvbG91cl9tYW51YWwodmFsdWVzID0gc3BlY2llc19jb2xzKSArIAogIHRoZW1lX21hdHQoKSArIAogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICJyaWdodCIpCmBgYAoKYGBge3IsIGN0bWF4LWZlY3VuZGl0eSwgZmlnLndpZHRoPTEwLCBmaWcuaGVpZ2h0PTd9CmdncGxvdChjdG1heF9yZXNpZHMsIGFlcyh5ID0gcmVzaWRzLCB4ID0gZmVjdW5kaXR5LCBjb2xvdXIgPSBzcF9uYW1lKSkgKyAKICBnZW9tX3Ntb290aChtZXRob2QgPSAibG0iLCBzZSA9IEYsIGxpbmV3aWR0aCA9IDIpICsgCiAgZ2VvbV9wb2ludChzaXplID0gMiwgYWxwaGEgPSAwLjUpICsgCiAgbGFicyh5ID0gIkNUbWF4ICjCsEMpIiwgCiAgICAgICB4ID0gIkZlY3VuZGl0eSAoIyBFZ2dzKSIpICsgCiAgc2NhbGVfY29sb3VyX21hbnVhbCh2YWx1ZXMgPSBzcGVjaWVzX2NvbHMpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKYGBgCgpgYGB7cn0KaWYocHJlZGljdF92dWxuID09IEYpewogIGtuaXRyOjprbml0X2V4aXQoKQp9CmBgYAoKCiMjIFByZWRpY3RpbmcgVnVsbmVyYWJpbGl0eSAKVXNpbmcgdGhlIG9ic2VydmVkIHRoZXJtYWwgbGltaXQgZGF0YSwgd2UgY2FuIHByb2R1Y2UgYSBoaW5kY2FzdCBvZiB0aGVybWFsIHN0cmVzcyBmb3IgTGFrZSBDaGFtcGxhaW4gY29wZXBvZHMuIEZvciB0aGVzZSBpbml0aWFsIGFzc2F5cywgd2Ugd2lsbCBkZWZpbmUgdGhlcm1hbCBzdHJlc3MgYXMgYW55IHRpbWUgd2hlbiBtYXhpbXVtIGRhaWx5IHdhdGVyIHRlbXBlcmF0dXJlIGlzIHdpdGhpbiAywrBDIG9mIGNvcGVwb2QgQ1RtYXggb3IgaGlnaGVyLiBXZSB3aWxsIHVzZSB0aHJlZSBkaWZmZXJlbnQgc2NlbmFyaW9zOiAxKSB0aGUgYXZlcmFnZSBDVG1heCBmb3IgZWFjaCBzcGVjaWVzLCAyKSBDVG1heCBwcmVkaWN0ZWQgdXNpbmcgY29sbGVjdGlvbiB0ZW1wZXJhdHVyZXMsIGFuZCAzKSBmb3Igc3BlY2llcyB0aGF0IGhhdmUgc3VmZmljaWVudCBkYXRhLCBDVG1heCBwcmVkaWN0ZWQgdXNpbmcgd2hpY2hldmVyIGVudmlyb25tZW50YWwgZmFjdG9yIGlzIHRoZSBzdHJvbmdlc3QgY2FuZGlkYXRlIGZvciBkcml2aW5nIGFjY2xpbWF0aW9uLiBJbiBhbGwgY2FzZXMsIGRhdGEgaXMgZmlsdGVyZWQgdG8ganVzdCB0aGVybWFsIGxpbWl0cyBvZiBhZHVsdCBmZW1hbGVzLiAKCiMjIyBTY2VuYXJpbyAxCmBgYHtyfQptZWFuX2N0bWF4ID0gZnVsbF9kYXRhICU+JSAKICBmaWx0ZXIoc2V4ID09ICJmZW1hbGUiKSAlPiUgIAogIGdyb3VwX2J5KHNwX25hbWUpICU+JSAKICBzdW1tYXJpemUoIm1lYW5fY3RtYXgiID0gbWVhbihjdG1heCkpICU+JSAKICBhcnJhbmdlKG1lYW5fY3RtYXgpCgprbml0cjo6a2FibGUobWVhbl9jdG1heCkKYGBgCgpgYGB7cn0KIyAjIENvbnN0cnVjdHMgdGhlIFVSTCBmb3IgdGhlIGZ1bGwgdGVtcGVyYXR1cmUgZGF0YSBzZXQ7IFJVTiBUSElTIE9OQ0UKIyBoaW5kX3VybCA9IGNvbnN0cnVjdE5XSVNVUkwoc2l0ZU51bWJlcnMgPSBzaXRlTnVtYmVyLCBwYXJhbWV0ZXJDZCA9IHBhcmFtZXRlckNkLCBzZXJ2aWNlID0gInV2IikKIyAKIyBoaW5kX3RlbXBfZGF0YSA9IGltcG9ydFdhdGVyTUwxKGhpbmRfdXJsLCBhc0RhdGVUaW1lID0gVCkgJT4lCiMgICBtdXRhdGUoImRhdGUiID0gYXMuRGF0ZShkYXRlVGltZSkpICU+JQojICAgc2VsZWN0KGRhdGUsICJ0ZW1wIiA9IFhfMDAwMTBfMDAwMDApCiMgCiMgd3JpdGUudGFibGUoeCA9IGhpbmRfdGVtcF9kYXRhLCBmaWxlID0gImhpbmRjYXN0X3RlbXBzLmNzdiIsIHJvdy5uYW1lcyA9IEYsIHNlcCA9ICIsIikKYGBgCgpgYGB7cn0KIyBnZ3Bsb3QoaGluZF90ZW1wX2RhdGEsIGFlcyh4ID0gZGF0ZSwgeSA9IHRlbXApKSArIAojICAgZ2VvbV9saW5lKGxpbmV3aWR0aCA9IDAuMSkgKyAKIyAgIGxhYnMoeCA9ICJEYXRlIiwgCiMgICAgICAgIHkgPSAiV2F0ZXIgVGVtcGVyYXR1cmUgKMKwQykiKSArCiMgICB0aGVtZV9tYXR0KCkKYGBgCgpJbiB0aGUgc2ltcGxlc3Qgc2NlbmFyaW8sIHNwZWNpZXMgdGhlcm1hbCBsaW1pdHMgYXJlIHN0YXRpYyB0aHJvdWdoIHRpbWUsIHJlcHJlc2VudGVkIGJ5IHRoZSBhdmVyYWdlIENUbWF4IG9mIGFkdWx0IGZlbWFsZSBjb3BlcG9kcy4gSW4gdGhpcyBzY2VuYXJpbywgb25seSB0aHJlZSBvZiB0aGUgc2V2ZW4gb2JzZXJ2ZWQgc3BlY2llcyBhcmUgZXhwb3NlZCB0byB0aGVybWFsIHN0cmVzcyAodGVtcGVyYXR1cmVzIHdpdGhpbiA1wrBDIG9mIENUbWF4KS4gVGVtcGVyYXR1cmVzIGFwcHJvYWNoZWQgdGhlIHRoZXJtYWwgbGltaXQgb2YgKkxlcHRvZGlhcHRvbXVzIHNpY2lsaXMqIG9uIGEgaGFuZGZ1bCBvZiBkYXlzLiBCeSBjb250cmFzdCwgKlNlbmVjZWxsYSBjYWxhbm9pZGVzKiBhbmQgKkxpbW5vY2FsYW51cyBtYWNydXJ1cyogd2VyZSBib3RoIGV4cG9zZWQgdG8gc3Vic3RhbnRpYWwgdGhlcm1hbCBzdHJlc3MgdGhyb3VnaG91dCBhIGxhcmdlIHBvcnRpb24gb2YgdGhlIHllYXIsIGxpa2VseSBleHBsYWluaW5nIHdoeSB0aGVzZSBzcGVjaWVzIGFyZSBhYnNlbnQgZnJvbSB0aGUgY29tbXVuaXR5IGZvciB0aGUgc3VtbWVyIGFuZCBmYWxsIHBlcmlvZHMuIAoKYGBge3IgZmlnLndpZHRoPTEwLCBmaWcuaGVpZ2h0PTV9CmhpbmQxX2RhdGEgPSBoaW5kX3RlbXBfZGF0YSAlPiUgCiAgZ3JvdXBfYnkoZGF0ZSkgJT4lIAogIHN1bW1hcml6ZSgiZGFpbHlfbWF4IiA9IG1heCh0ZW1wKSwKICAgICAgICAgICAgImRhaWx5X21lYW4iID0gbWVhbih0ZW1wKSwpICU+JSAKICBiaW5kX2NvbHMocGl2b3Rfd2lkZXIobWVhbl9jdG1heCwgbmFtZXNfZnJvbSA9IHNwX25hbWUsIHZhbHVlc19mcm9tID0gbWVhbl9jdG1heCkpICU+JSAgCiAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKC1kYXRlLCAtZGFpbHlfbWF4LCAtZGFpbHlfbWVhbiksCiAgICAgICAgICAgICAgIG5hbWVzX3RvID0gInNwZWNpZXMiLCAKICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gIm1lYW5fY3RtYXgiKSAlPiUgIAogIG11dGF0ZShsaW1fZGlmZiA9IG1lYW5fY3RtYXggLSBkYWlseV9tYXgpICU+JSAgCiAgbXV0YXRlKGRveSA9IHlkYXkoZGF0ZSksCiAgICAgICAgICJtZXRob2QiID0gIk5vX2FjY2xpbWF0aW9uIikKCmhpbmRfZGFpbHlfdGVtcF9kYXRhID0gaGluZF90ZW1wX2RhdGEgJT4lCiAgdW5ncm91cCgpICU+JSAKICBncm91cF9ieShkYXRlKSAlPiUgCiAgc3VtbWFyaXNlKG1lYW5fdGVtcCA9IG1lYW4odGVtcCksCiAgICAgICAgICAgIG1lZF90ZW1wID0gbWVkaWFuKHRlbXApLAogICAgICAgICAgICB2YXJfdGVtcCA9IHZhcih0ZW1wKSwgCiAgICAgICAgICAgIG1pbl90ZW1wID0gbWluKHRlbXApLCAKICAgICAgICAgICAgbWF4X3RlbXAgPSBtYXgodGVtcCkpICU+JSAKICBtdXRhdGUoInJhbmdlX3RlbXAiID0gbWF4X3RlbXAgLSBtaW5fdGVtcCkKCiN0YWJsZShoaW5kMV9kYXRhJHNwZWNpZXMpCgpoaW5kMV9kYXRhICU+JSAKICBmaWx0ZXIobGltX2RpZmYgPD0gNSkgJT4lICAKICBnZ3Bsb3QoYWVzKHggPSBkb3ksIHkgPSBsaW1fZGlmZiwgY29sb3VyID0gc3BlY2llcykpICsKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSAwKSArIAogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IDUsIAogICAgICAgICAgICAgY29sb3VyID0gImdyZXkiKSArIAogIGdlb21fcG9pbnQoYWxwaGEgPSAwLjUpICsKICBnZW9tX3Ntb290aCgpICsgCiAgbGFicyh4ID0gIkRheSBvZiBZZWFyIiwgCiAgICAgICB5ID0gIlByZWRpY3RlZCBXYXJtaW5nIFRvbGVyYW5jZSBcbijCsEMgQWJvdmUgRGFpbHkgTWF4KSIpICsgCiAgdGhlbWVfbWF0dCgpICsgCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gInJpZ2h0IikKYGBgCgojIyMgU2NlbmFyaW8gMgpJbiB0aGUgc2Vjb25kIHNjZW5hcmlvLCB0aGVybWFsIGxpbWl0cyB2YXJ5IHdpdGhpbiBhbmQgYmV0d2VlbiBzcGVjaWVzLiBBIHNpbXBsZSBtb2RlbCBpcyB1c2VkIHRvIHByZWRpY3Qgc3BlY2llcyB0aGVybWFsIGxpbWl0cyBiYXNlZCBvbiBtZWFuIGRhaWx5IHRlbXBlcmF0dXJlIChDVG1heCBhcyBhIGZ1bmN0aW9uIG9mIHNwZWNpZXMgYW5kIGNvbGxlY3Rpb24gdGVtcGVyYXR1cmUsIGJ1dCB3aXRob3V0IHRoZSBpbnRlcmFjdGlvbiBiZXR3ZWVuIHRoZXNlIHR3byBmYWN0b3JzKS4gVGhlc2UgcHJlZGljdGVkIHRoZXJtYWwgbGltaXRzIGFyZSB0aGVuIGNvbXBhcmVkIGFnYWluc3QgdGhlIG1heGltdW0gZGFpbHkgdGVtcGVyYXR1cmUgdG8gZXN0aW1hdGUgdGhlcm1hbCBzdHJlc3MsIGFzIGluIFNjZW5hcmlvIDEuIEluY2x1ZGluZyB0aGlzIHNpbXBsZSBmb3JtIG9mIGFjY2xpbWF0aW9uIGluIHRoZSBtb2RlbCByZWR1Y2VkIHRoZSBkZWdyZWUgb2YgdGhlcm1hbCBzdHJlc3MgZm9yIGVhY2ggc3BlY2llcywgZWxpbWluYXRpbmcgaXQgZW50aXJlbHkgZm9yICpMZXB0b2RpYXB0b211cyBzaWNpbGlzKi4gTm90ZSB0aGF0IHRoZSBtYWduaXR1ZGUgb2YgdGhlIHByZWRpY3RlZCBzdHJlc3MgaXMgIGxvdyBlbm91Z2ggdGhhdCByZW1vdmluZyB0aGUgNcKwQyBidWZmZXIgYXJvdW5kIHRoZSBwcmVkaWN0ZWQgdGhlcm1hbCBsaW1pdHMgd291bGQgYWN0dWFsbHkgbGltaXQgcHJlZGljdGVkIHRoZXJtYWwgc3RyZXNzIHRvIGp1c3QgYSBmZXcgZGF5cyBmb3IgKlNlbmVjZWxsYSBjYWxhbm9pZGVzKi4gCgpgYGB7ciBmaWcuaGVpZ2h0PTUsIGZpZy53aWR0aD0xMH0KaGluZGNhc3RfbW9kZWwxID0gbG0oZGF0YSA9IGZpbHRlcihmdWxsX2RhdGEsIHNleCA9PSAiZmVtYWxlIiksCiAgICAgICAgICAgICAgICAgICAgIGN0bWF4IH4gY29sbGVjdGlvbl90ZW1wICsgc3BfbmFtZSkKCmhpbmQyX2RhdGEgPSBoaW5kX3RlbXBfZGF0YSAlPiUgCiAgZ3JvdXBfYnkoZGF0ZSkgJT4lIAogIHN1bW1hcml6ZSgiY29sbGVjdGlvbl90ZW1wIiA9IG1lYW4odGVtcCksCiAgICAgICAgICAgICJkYWlseV9tYXgiID0gbWF4KHRlbXApKSAlPiUgCiAgYmluZF9jb2xzKAogICAgcGl2b3Rfd2lkZXIobWVhbl9jdG1heCwgCiAgICAgICAgICAgICAgICBuYW1lc19mcm9tID0gc3BfbmFtZSwgCiAgICAgICAgICAgICAgICB2YWx1ZXNfZnJvbSA9IG1lYW5fY3RtYXgpKSAlPiUgCiAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKC1kYXRlLCAtZGFpbHlfbWF4LCAtY29sbGVjdGlvbl90ZW1wKSwKICAgICAgICAgICAgICAgbmFtZXNfdG8gPSAic3BfbmFtZSIsIAogICAgICAgICAgICAgICB2YWx1ZXNfdG8gPSAibWVhbl9jdG1heCIpICU+JSAKICBzZWxlY3QoLW1lYW5fY3RtYXgpICU+JSAKICBtdXRhdGUoInByZWRfY3RtYXgiID0gcHJlZGljdC5sbSAoaGluZGNhc3RfbW9kZWwxLCBuZXdkYXRhID0gLikpICU+JSAKICBzZWxlY3QoZGF0ZSwgImRhaWx5X21lYW4iID0gY29sbGVjdGlvbl90ZW1wLCBkYWlseV9tYXgsICJzcGVjaWVzIiA9IHNwX25hbWUsIHByZWRfY3RtYXgpICU+JSAKICBtdXRhdGUobGltX2RpZmYgPSBwcmVkX2N0bWF4IC0gZGFpbHlfbWF4KSAlPiUgCiAgI2ZpbHRlcihsaW1fZGlmZiA8PSAwKSAlPiUgIAogIG11dGF0ZShkb3kgPSB5ZGF5KGRhdGUpLAogICAgICAgICAibWV0aG9kIiA9ICJDb25zdGFudF9hY2NsaW1hdGlvbiIpCgojIGdncGxvdChoaW5kMl9kYXRhLCBhZXMoeCA9IGRhaWx5X21lYW4sIHkgPSBwcmVkX2N0bWF4LCBjb2xvdXIgPSBzcGVjaWVzKSkgKwojICAgZ2VvbV9zbW9vdGgobWV0aG9kID0gImxtIikgCgojIHRhYmxlKGhpbmQyX2RhdGEkc3BlY2llcykKaGluZDJfZGF0YSAlPiUgIAogIGZpbHRlcihsaW1fZGlmZiA8PSA1KSAlPiUgIAogIGdncGxvdChhZXMoeCA9IGRveSwgeSA9IGxpbV9kaWZmLCBjb2xvdXIgPSBzcGVjaWVzKSkgKwogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IDApICsgCiAgZ2VvbV9obGluZSh5aW50ZXJjZXB0ID0gNSwgCiAgICAgICAgICAgICBjb2xvdXIgPSAiZ3JleSIpICsgCiAgZ2VvbV9wb2ludChhbHBoYSA9IDAuNSkgKwogIGdlb21fc21vb3RoKCkgKyAKICBsYWJzKHggPSAiRGF5IG9mIFllYXIiLCAKICAgICAgIHkgPSAiUHJlZGljdGVkIFdhcm1pbmcgVG9sZXJhbmNlIFxuKMKwQyBBYm92ZSBEYWlseSBNYXgpIikgKyAKICB0aGVtZV9tYXR0KCkgKyAKICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAicmlnaHQiKQpgYGAKCiMjIyBTY2VuYXJpbyAzClRoZSBmaW5hbCBzY2VuYXJpbyBhbGxvd3MgdGhlIGVudmlyb25tZW50YWwgdmFyaWFibGUgdXNlZCB0byBwcmVkaWN0IENUbWF4IHRvIHZhcnkgYmV0d2VlbiBzcGVjaWVzLiBGb3Igc3BlY2llcyBvYnNlcnZlZCBpbiBmZXdlciB0aGFuIDUgY29sbGVjdGlvbnMsIHdlIHVzZSB0aGUgc2FtZSBhcHByb2FjaCBhcyBpbiBTY2VuYXJpbyAyLiBGb3Igc3BlY2llcyBvYnNlcnZlZCBpbiBtb3JlIHRoYW4gNSBjb2xsZWN0aW9ucywgaG93ZXZlciwgdGhlIGZhY3RvciB3aXRoIHRoZSBzdHJvbmdlc3QgY29ycmVsYXRpb24gd2l0aCBDVG1heCBpcyB1c2VkIHRvIHByZWRpY3QgdGhlcm1hbCBsaW1pdHMuIFRoZXNlIGZhY3RvcnMgYXJlIGluY2x1ZGVkIGJlbG93LgoKYGBge3J9CmhpbmRfcHJlZHMgPSBjb3JyX3ZhbHMgJT4lICAKICBmaWx0ZXIoc2lnID09ICJTaWcuIikgJT4lIAogIGRyb3BfbmEoY29ycmVsYXRpb24pICU+JSAKICBncm91cF9ieShzcF9uYW1lKSAlPiUKICBhcnJhbmdlKGRlc2MoY29ycmVsYXRpb24pKSAlPiUgCiAgc2xpY2VfaGVhZChuID0gMSkgJT4lIAogIHNlbGVjdCgiU3BlY2llcyIgPSBzcF9uYW1lLCAiUHJlZGljdG9yIiA9IHBhcmFtZXRlciwgIkR1cmF0aW9uIiA9IGR1cmF0aW9uLCAiQ29ycmVsYXRpb24iID0gY29ycmVsYXRpb24sICJQLVZhbHVlIiA9IHAudmFsdWUpCgprbml0cjo6a2FibGUoaGluZF9wcmVkcywgYWxpZ24gPSAiYyIpCmBgYAoKYGBge3J9CmhpbmQzX2RhdGEgPSBoaW5kMl9kYXRhICU+JSAjIENvbnRhaW5zIGRhdGEgZm9yIHNwZWNpZXMgdGhhdCB3b24ndCBjaGFuZ2UgZnJvbSBzY2VuYXJpbyAyCiAgZmlsdGVyKCEoc3BlY2llcyAlaW4lIGNvcnJfdmFscyRzcF9uYW1lKSkKCnByZWRzX3RvX3B1bGwgPSBoaW5kX3ByZWRzICU+JSAgCiAgc2VsZWN0KFNwZWNpZXMsIFByZWRpY3RvciwgRHVyYXRpb24pIAoKZm9yKGkgaW4gMTpsZW5ndGgocHJlZHNfdG9fcHVsbCRTcGVjaWVzKSl7CiAgCiAgaWYocHJlZHNfdG9fcHVsbCREdXJhdGlvbltpXSA9PSAicHJpb3IiKXsgI1RoZSBwcmlvciBkYXkgdGVtcGVyYXR1cmUgbWV0cmljcyBzaG91bGQgYmUgdXNlZAogICAgZHVyYXRpb24gPSBOQQogICAgCiAgICBwcmVkaWN0b3JzID0gaGluZF9kYWlseV90ZW1wX2RhdGEgJT4lIAogICAgICBtdXRhdGUoZGF0ZSA9IGRhdGUgKyAxKSAKICAgIAogICAgcGFyYW1ldGVyID0gcHJlZHNfdG9fcHVsbCRQcmVkaWN0b3JbaV0KICAgIAogICAgbW9kZWxfZGF0YSA9IGZ1bGxfZGF0YSAlPiUKICAgICAgZmlsdGVyKHNwX25hbWUgJWluJSBwcmVkc190b19wdWxsJFNwZWNpZXNbaV0pICU+JSAKICAgICAgZmlsdGVyKHNleCA9PSAiZmVtYWxlIikgJT4lIAogICAgICBtdXRhdGUoY29sbGVjdGlvbl9kYXRlID0gYXNfZGF0ZShjb2xsZWN0aW9uX2RhdGUpKSAlPiUgCiAgICAgIGlubmVyX2pvaW4ocHJlZGljdG9ycywgam9pbl9ieShjb2xsZWN0aW9uX2RhdGUgPT0gZGF0ZSkpICU+JSAgCiAgICAgIHNlbGVjdChjdG1heCwgY29udGFpbnMocGFyYW1ldGVyKSkKICAgIAogICAgaWYoZGltKG1vZGVsX2RhdGEpWzJdID09IDIpewogICAgICBoaW5kLm1vZGVsID0gbG0oZGF0YSA9IG1vZGVsX2RhdGEsIAogICAgICAgICAgICAgICAgICAgICAgY3RtYXggfiAuKQogICAgICAKICAgICAgc3BfZGF0YSA9IHByZWRpY3RvcnMgJT4lIAogICAgICAgIHNlbGVjdChkYXRlLCBjb250YWlucyhwYXJhbWV0ZXIpKSAlPiUgCiAgICAgICAgbXV0YXRlKHByZWRfY3RtYXggPSBwcmVkaWN0KGhpbmQubW9kZWwsIG5ld2RhdGEgPSAuKSkgJT4lICAKICAgICAgICBzZWxlY3QoZGF0ZSwgcHJlZF9jdG1heCkgJT4lIAogICAgICAgIGlubmVyX2pvaW4oaGluZF9kYWlseV90ZW1wX2RhdGEsIGJ5ID0gYygiZGF0ZSIpKSAlPiUgCiAgICAgICAgbXV0YXRlKCJzcGVjaWVzIiA9IHByZWRzX3RvX3B1bGwkU3BlY2llc1tpXSwKICAgICAgICAgICAgICAgImRveSIgPSB5ZGF5KGRhdGUpLAogICAgICAgICAgICAgICBsaW1fZGlmZiA9IHByZWRfY3RtYXggLSBtYXhfdGVtcCkgJT4lIAogICAgICAgIHNlbGVjdChkYXRlLCBkYWlseV9tZWFuID0gbWVhbl90ZW1wLCBkYWlseV9tYXggPSBtYXhfdGVtcCwgc3BlY2llcywgcHJlZF9jdG1heCwgbGltX2RpZmYsIGRveSkKICAgICAgCiAgICAgIGhpbmQzX2RhdGEgPSBiaW5kX3Jvd3MoaGluZDNfZGF0YSwgc3BfZGF0YSkKICAgIH1lbHNlewogICAgICBwcmludCgiVG9vIG1hbnkgY29sdW1ucyBzZWxlY3RlZCIpCiAgICB9CiAgICAKICAgIAogIH1lbHNlewogICAgZHVyYXRpb24gPSBhcy5udW1lcmljKHByZWRzX3RvX3B1bGwkRHVyYXRpb25baV0pCiAgfQogIAogIGlmKHByZWRzX3RvX3B1bGwkRHVyYXRpb25baV0gIT0gInByaW9yIiAmIGlzLm5hKGR1cmF0aW9uKSl7ICNEYWlseSB0ZW1wZXJhdHVyZXMgc2hvdWxkIGJlIHVzZWQsIGFzIGluIFNjZW5hcmlvIDIKICAgIHNwX2RhdGEgPSBoaW5kMl9kYXRhICU+JSAKICAgICAgZmlsdGVyKHNwZWNpZXMgPT0gcHJlZHNfdG9fcHVsbCRTcGVjaWVzW2ldKQogICAgCiAgICBoaW5kM19kYXRhID0gYmluZF9yb3dzKGhpbmQzX2RhdGEsIHNwX2RhdGEpCiAgfQogIAogIGlmKGlzLm51bWVyaWMoZHVyYXRpb24pKXsKICAgICNOZWl0aGVyIHRoZSBwcmlvciBkYXkgbm9yIGRheSBvZiBtZXRyaWNzIHNob3VsZCBiZSB1c2VkOyB1c2UgZHVyYXRpb24gYXMgbl9kYXlzCiAgICAKICAgIHByZWRpY3RvcnMgPSBnZXRfcHJlZGljdG9ycyhkYWlseV92YWx1ZXMgPSBoaW5kX2RhaWx5X3RlbXBfZGF0YSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmF3X3RlbXAgPSBoaW5kX3RlbXBfZGF0YSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbl9kYXlzID0gZHVyYXRpb24pCiAgICAKICAgIHBhcmFtZXRlciA9IHByZWRzX3RvX3B1bGwkUHJlZGljdG9yW2ldCiAgICAKICAgIG1vZGVsX2RhdGEgPSBmdWxsX2RhdGEgJT4lCiAgICAgIGZpbHRlcihzcF9uYW1lICVpbiUgcHJlZHNfdG9fcHVsbCRTcGVjaWVzW2ldKSAlPiUgCiAgICAgIGZpbHRlcihzZXggPT0gImZlbWFsZSIpICU+JSAKICAgICAgbXV0YXRlKGNvbGxlY3Rpb25fZGF0ZSA9IGFzX2RhdGUoY29sbGVjdGlvbl9kYXRlKSkgJT4lIAogICAgICBpbm5lcl9qb2luKHByZWRpY3RvcnMsIGpvaW5fYnkoY29sbGVjdGlvbl9kYXRlID09IGRhdGUpKSAlPiUgIAogICAgICBzZWxlY3QoY3RtYXgsIGNvbnRhaW5zKHBhc3RlKCJkYXlfIiwgcGFyYW1ldGVyLCBzZXAgPSAiIikpKQogICAgCiAgICBpZihkaW0obW9kZWxfZGF0YSlbMl0gPT0gMil7CiAgICAgIGhpbmQubW9kZWwgPSBsbShkYXRhID0gbW9kZWxfZGF0YSwgCiAgICAgICAgICAgICAgICAgICAgICBjdG1heCB+IC4pCiAgICAgIAogICAgICBzcF9kYXRhID0gcHJlZGljdG9ycyAlPiUgCiAgICAgICAgc2VsZWN0KGRhdGUsIGNvbnRhaW5zKHBhcmFtZXRlcikpICU+JSAKICAgICAgICBtdXRhdGUocHJlZF9jdG1heCA9IHByZWRpY3QoaGluZC5tb2RlbCwgbmV3ZGF0YSA9IC4pKSAlPiUgIAogICAgICAgIHNlbGVjdChkYXRlLCBwcmVkX2N0bWF4KSAlPiUgCiAgICAgICAgaW5uZXJfam9pbihoaW5kX2RhaWx5X3RlbXBfZGF0YSwgYnkgPSBjKCJkYXRlIikpICU+JSAKICAgICAgICBtdXRhdGUoInNwZWNpZXMiID0gcHJlZHNfdG9fcHVsbCRTcGVjaWVzW2ldLAogICAgICAgICAgICAgICAiZG95IiA9IHlkYXkoZGF0ZSksCiAgICAgICAgICAgICAgIGxpbV9kaWZmID0gcHJlZF9jdG1heCAtIG1heF90ZW1wKSAlPiUgCiAgICAgICAgc2VsZWN0KGRhdGUsIGRhaWx5X21lYW4gPSBtZWFuX3RlbXAsIGRhaWx5X21heCA9IG1heF90ZW1wLCBzcGVjaWVzLCBwcmVkX2N0bWF4LCBsaW1fZGlmZiwgZG95KQogICAgICAKICAgICAgaGluZDNfZGF0YSA9IGJpbmRfcm93cyhoaW5kM19kYXRhLCBzcF9kYXRhKQogICAgICAKICAgIH1lbHNlewogICAgICBwcmludCgiVG9vIG1hbnkgY29sdW1ucyBzZWxlY3RlZCIpCiAgICB9CiAgICAKICB9Cn0KCmhpbmQzX2RhdGEgPSBoaW5kM19kYXRhICU+JSAKICBtdXRhdGUoIm1ldGhvZCIgPSAiVmFyaWFibGVfYWNjbGltYXRpb24iKQpgYGAKClRoaXMgdGhpcmQgYXBwcm9hY2ggZGlkIG5vdCBhZmZlY3QgdGhlIHByZWRpY3RlZCBwYXR0ZXJucyBpbiAqTGltbm9jYWxhbnVzKiBvciAqU2VuZWNlbGxhKiAobmVpdGhlciBzcGVjaWVzIGhhcyBiZWVuIG9ic2VydmVkIGluIGVub3VnaCBjb2xsZWN0aW9ucyB0byBlc3RpbWF0ZSB0aGUgZWZmZWN0cyBvZiBkaWZmZXJlbnQgZW52aXJvbm1lbnRhbCBmYWN0b3JzKS4gQ2hhbmdpbmcgdGhlIGFjY2xpbWF0aW9uIGFwcHJvYWNoIGRpZCBhZmZlY3QgcGF0dGVybnMgaW4gdGhlcm1hbCBsaW1pdHMgaW4gdGhlIG90aGVyIHNwZWNpZXMgdGhvdWdoLiBUaGUgZmlndXJlIGJlbG93IHNob3dzIGhvdyBwcmVkaWN0ZWQgd2FybWluZyB0b2xlcmFuY2UgdmFyaWVzIG92ZXIgdGhlIHllYXIgaW4gdGhlIHNldmVuIHNwZWNpZXMsIGJhc2VkIG9uIHRoZSB0aHJlZSBkaWZmZXJlbnQgcHJlZGljdGlvbiBtZXRob2RzLiBJbiBnZW5lcmFsLCBjb25zdGFudCB0aGVybWFsIGxpbWl0cyAodGhlICdubyBhY2NsaW1hdGlvbicgbWV0aG9kKSByZXN1bHRlZCBpbiBsYXJnZXIgd2FybWluZyB0b2xlcmFuY2UgZHVyaW5nIHRoZSB3aW50ZXIgYW5kIGxvd2VyIHdhcm1pbmcgdG9sZXJhbmNlIGR1cmluZyB0aGUgc3VtbWVyLCBhbHRob3VnaCB0aGlzIGVmZmVjdCB3YXMgc21hbGwgaW4gbW9zdCBzcGVjaWVzLiAgICAgCgpgYGB7ciBmaWcud2lkdGg9MTAsIGZpZy5oZWlnaHQ9MTB9CnN5bnRoZXNpcyA9IGJpbmRfcm93cygKICBzZWxlY3QoaGluZDFfZGF0YSwgZGF0ZSwgZG95LCBkYWlseV9tZWFuLCBkYWlseV9tYXgsIHNwZWNpZXMsICJwcmVkX2N0bWF4IiA9IG1lYW5fY3RtYXgsIGxpbV9kaWZmLCBtZXRob2QpLAogIHNlbGVjdChoaW5kMl9kYXRhLCBkYXRlLCBkb3ksIGRhaWx5X21lYW4sIGRhaWx5X21heCwgIHNwZWNpZXMsIHByZWRfY3RtYXgsIGxpbV9kaWZmLCBtZXRob2QpLAogIHNlbGVjdChoaW5kM19kYXRhLCBkYXRlLCBkb3ksIGRhaWx5X21lYW4sIGRhaWx5X21heCwgIHNwZWNpZXMsIHByZWRfY3RtYXgsIGxpbV9kaWZmLCBtZXRob2QpKSAlPiUgCiAgbXV0YXRlKG1ldGhvZCA9IGZjdF9yZWxldmVsKG1ldGhvZCwgIk5vX2FjY2xpbWF0aW9uIiwgIkNvbnN0YW50X2FjY2xpbWF0aW9uIiwgIlZhcmlhYmxlX2FjY2xpbWF0aW9uIikpCgpjbGltYXRvbG9neSA9IHN5bnRoZXNpcyAlPiUgCiAgZ3JvdXBfYnkoc3BlY2llcywgZG95LCBtZXRob2QpICU+JSAgCiAgc3VtbWFyaXNlKCJtZWFuX2RpZmYiID0gbWVhbihsaW1fZGlmZiksCiAgICAgICAgICAgICJtaW5fZGlmZiIgPSBtaW4obGltX2RpZmYpLAogICAgICAgICAgICAibWF4X2RpZmYiID0gbWF4KGxpbV9kaWZmKSkgJT4lIAogIG11dGF0ZShtZXRob2QgPSBmY3RfcmVsZXZlbChtZXRob2QsICJOb19hY2NsaW1hdGlvbiIsICJDb25zdGFudF9hY2NsaW1hdGlvbiIsICJWYXJpYWJsZV9hY2NsaW1hdGlvbiIpKQoKYWNjX2VmZmVjdHMgPSBzeW50aGVzaXMgJT4lIAogIHBpdm90X3dpZGVyKGlkX2NvbHMgPSBjKGRhdGUsIHNwZWNpZXMsIGRveSksIAogICAgICAgICAgICAgIG5hbWVzX2Zyb20gPSBtZXRob2QsIAogICAgICAgICAgICAgIHZhbHVlc19mcm9tID0gbGltX2RpZmYpICU+JSAgCiAgbXV0YXRlKCJjb25zdF9hY2NfZWZmZWN0IiA9IENvbnN0YW50X2FjY2xpbWF0aW9uIC0gTm9fYWNjbGltYXRpb24sCiAgICAgICAgICJ2YXJfYWNjX2VmZmVjdCIgPSBWYXJpYWJsZV9hY2NsaW1hdGlvbiAtIE5vX2FjY2xpbWF0aW9uKQoKZ2dwbG90KHN5bnRoZXNpcywgYWVzKHggPSBkb3ksIHkgPSBsaW1fZGlmZiwgY29sb3VyID0gbWV0aG9kKSkgKyAKICBmYWNldF93cmFwKHNwZWNpZXN+LikgKyAKICBnZW9tX2hsaW5lKHlpbnRlcmNlcHQgPSAwKSArIAogIGdlb21faGxpbmUoeWludGVyY2VwdCA9IDUsIGNvbG91ciA9ICJncmV5IikgKyAKICBnZW9tX3BvaW50KGFscGhhID0gMC4xKSArIAogIGxhYnMoeCA9ICJEYXkgb2YgWWVhciIsIAogICAgICAgeSA9ICJQcmVkaWN0ZWQgV2FybWluZyBUb2xlcmFuY2UgKMKwQyBBYm92ZSBEYWlseSBNYXgpIikgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cyhiYXNlX3NpemUgPSAxOCkgKyAKICB0aGVtZShzdHJpcC50ZXh0LngudG9wID0gZWxlbWVudF90ZXh0KHNpemUgPSAxMCkpCmBgYAoKYGBge3IsIGluY2x1ZGUgPSBGfQp5ZWFybHlfc3VtbWFyeSA9IHN5bnRoZXNpcyAlPiUgIAogIG11dGF0ZSgieWVhciIgPSB5ZWFyKGRhdGUpKSAlPiUgCiAgZ3JvdXBfYnkoc3BlY2llcywgeWVhciwgbWV0aG9kKSAlPiUgCiAgICBzdW1tYXJpc2UoIm1pbl93dCIgPSBtaW4obGltX2RpZmYpLAogICAgICAgICAgICAgICJtYXhfd3QiID0gbWF4KGxpbV9kaWZmKSkgJT4lIAogIHBpdm90X2xvbmdlcihjb2xzID0gYyhtaW5fd3QsIG1heF93dCksIAogICAgICAgICAgICAgICBuYW1lc190byA9ICJtZXRyaWMiLCAKICAgICAgICAgICAgICAgdmFsdWVzX3RvID0gInd0IikKCmdncGxvdCh5ZWFybHlfc3VtbWFyeSwgYWVzKHggPSBtZXRob2QsIHkgPSB3dCwgY29sb3VyID0gbWV0cmljKSkgKyAKICBmYWNldF93cmFwKC5+c3BlY2llcykgKyAKICBnZW9tX3BvaW50KCkgKyAKICB0aGVtZV9tYXR0X2ZhY2V0cygpICsgCiAgdGhlbWUoYXhpcy50ZXh0LnggPSBlbGVtZW50X3RleHQoYW5nbGUgPSAzMDAsIGhqdXN0ID0gMCwgdmp1c3QgPSAwLjUpKQpgYGAKCg==